在shell脚本中调用另一个脚本的三种不同方法(fork, exec, source)

  • fork ( /directory/script.sh) :如果shell中包含执行命令,那么子命令并不影响父级的命令。

    在子命令执行完后再执行父级命令,子级的环境变量不会影响到父级。fork就是直接在脚本里面用/directory/script.sh来调用脚本。运行的时候开一个子shell执行调用的脚本,子shell执行的时候, parent-shell还在。子shell执行完毕后返回parent-shell. 子shell从parent-shell继承环境变量。但是子shell中的环境变量不会带回parent-shell。

  • exec (exec /directory/script.sh):执行子级的命令后,不再执行父级命令。

      exec与fork不同,不需要新开一个子shell来执行被调用的脚本。被调用的脚本与父脚本在同一个shell内执行。但是使用exec调用一个新脚本以后, 父脚本中exec行之后的内容就不会再执行了。这是exec和source的区别

  • source (source /directory/script.sh):执行子级命令后继续执行父级命令,同时子级设置的环境变量会影响到父级的环境变量。

与fork的区别是不新开一个子shell来执行被调用的脚本,而是在同一个shell中执行. 所以被调用的脚本中声明的变量和环境变量, 都可以在主脚本中得到和使用。

1.sh

#!/bin/bash
A=B 
echo "PID for 1.sh before exec/source/fork:$$"
export A
echo "1.sh: $A is $A"
case $1 in
        e)
            echo "using exec¡­"
            exec ./2.sh ;;
        s)
            echo "using source¡­"
            . ./2.sh ;;
        *)
            echo "using fork by default¡­"
            ./2.sh ;;
esac
echo "PID for 1.sh after exec/source/fork:$$"
echo "1.sh: $A is $A"

 2.sh

#!/bin/bash
echo "PID for 2.sh: $$"
echo "2.sh get $A=$A from 1.sh"
A=C
export A
echo "2.sh: $A is $A"

   exec 执行结果

sh  1.sh  e
PID for 1.sh before exec/source/fork:1945
1.sh: $A is B
using exec
PID for 2.sh: 1945
2.sh get $A=B from 1.sh
2.sh: $A is C

 source 执行结果

sh  1.sh  s
PID for 1.sh before exec/source/fork:1948
1.sh: $A is B
using source¡
PID for 2.sh: 1948
2.sh get $A=B from 1.sh
2.sh: $A is C
PID for 1.sh after exec/source/fork:1948
1.sh: $A is C

fork执行结果

sh  1.sh 
PID for 1.sh before exec/source/fork:1942
1.sh: $A is B
using fork by default
PID for 2.sh: 1943
2.sh get $A=B from 1.sh
2.sh: $A is C
PID for 1.sh after exec/source/fork:1942
1.sh: $A is B