在读《linux命令行与shell脚本编程大全》这本书的时候,第五章shell的父子关系,有这样一条命令很有趣
$ coproc ( sleep 10; sleep 2 )
coproc命令可以在后台生成一个子shell(sub or child?),并在这个子shell中执行命令。
而命令分组的一种,即进程列表(;)也会生成一个子shell(sub or child?)来执行对应命令。
将coproc与命令分组相结合,即可产生嵌套的子shell。
而环境变量BASH_SUBSHELL是记录一个Bash进程实例中多个子Shell(Sub shell)嵌套深度的累加器,与之相对的是
SHEVL,记录多个Bash进程实例嵌套深度的累加器。
$ ( echo $SHLVL; echo $BASH_SUBSHELL )
1
1
$ bash
$ ( echo $SHLVL; echo $BASH_SUBSHELL )
2
1
我们可以看到,进程列表(;)生成的实际上是subshell也就是真正的子shell,而bash命令对应生成的"子"shell实际上是child shell。
coproc在后台生成的即为 child shell。
从 c 语言层面讲,真正的子 Shell 是当前 Shell 进程调用了 fork() 函数,在内存中复制出一个几乎一模一样的子进程。而执行 bash 命令启动的所谓 child shell 是在执行 fork() 函数的基础上,又执行了一次 execute() 函数,execve() 函数会重新加载硬盘上的 bash 命令并执行,替换刚才 fork 出来的那个 shell 进程,除了传入的环境变量外,是个崭新的进程。