这是一个经常在面试时被问到的一个问题,对于刚刚接触shell的初学者来说,确实不太好搞明白这三者的区别,下面我通过两个脚本来帮助你理解它们。
1)我们所执行的任何程序,都是由父进程(parent process)所产生出来的一个子进程(child process),子进程在结束后,将返回到父进程去。此现像在Linux系统中被称为fork。当子进程被产生的时候,将会从父进程那里获得一定的资源分配、以及继承父进程的环境( 如环境变量)。
2)环境变量大体可以分为三类:
3)环境变量只能从父进程到子进程单向继承。换句话说:在子进程中的环境如何变更,均不会影响父进程的环境。
4)先准备两个示例脚本:
vi 1.sh ##内容如下
#!/bin/bash
A=aminglinux
echo "PID for 1.sh before exec/source/fork:$$"
export A
echo "1.sh: $A is $A"
case $1 in
fork)
echo "using fork"
bash 2.sh
;;
source)
echo "using source"
source 2.sh
;;
exec)
echo "using exec"
exec ./2.sh
;;
*)
echo "using fork"
bash 2.sh
;;
esac
echo "PID for 1.sh after exec/source/fork:$$"
echo "1.sh: $A is $A"
vi 2.sh ##内容如下
#!/bin/bash
echo "PID for 2.sh: $$"
echo "2.sh get $A=$A from 1.sh"
A=ops
export A
echo "2.sh: $A is $A"
给两个脚本执行权限
chmod +x 1.sh 2.sh
bash 1.sh fork
1)1.sh的PID为15242也就是父shell的PID,而2.sh的PID为15243,这个是子shell的PID。2)在1.sh里定义了变量A,值为aminglinux,然后fork了一个子shell去执行了2.sh,在2.sh里变量A的值是ops,但是当2.sh执行完后,再回到1.sh,变量A的值依然是aminglinux。
Source模式下,子shell执行时获取的环境变量会会影响到父shell。与fork的区别在于,不会额外打开一个sub-shell来执行被调用的脚本,而是在同一个shell中执行。所以,被调用的脚本中声明的变量和环境变量, 都可以在主脚本中得到和使用。
下面来执行下示例脚本:
bash 1.sh source
1)无论1.sh还是2.sh,PID都是17164,这说明source并不会开启sub-shell,而是和父shell使用了同一个进程。
2)source 2.sh后,变量A的值变成了ops,而后也被带到了1.sh里。
Exec模式下,一旦执行了子shell,就不会再去执行父shell了。它与fork不同,不需要新开一个sub-shell来执行被调用的脚本,被调用的脚本与父shell在同一个shell内执行,这个特性和source一样。但是使用exec调用一个新脚本后, 父shell中exec之后的内容就不会再执行了。
我们来看示例脚本执行结果:
bash 1.sh exec
1)1.sh和2.sh的PID都是18633,这说明exec和source一样,并不会开启sub-shell,而是和父shell使用了同一个进程。
2)exec调用完2.sh之后,脚本就结束了,没有再继续,这是exec的特性!
审核编辑 :李倩
全部0条评论
快来发表一下你的评论吧 !