准备知识
1.我们所执行的任何程序,都是由父进程(parent process)所产生出来的一个子进程(child process),子进程在结束后,将返回到父进程去。此一现像在Linux系统中被称为 fork。当子进程被产生的时候,将会从父进程那里获得一定的资源分配、及(更重要的是)继承父进程的环境﹗
2.Shell变量大致可以分为3种类型:
内部变量:系统提供,不用定义,不能修改,比如$#,$?,$*,$0等
环境变量:系统提供,不用定义,可以修改,当前进程及其子进程中使用,比如PATH,PWD,SHELL等
用户变量(本地变量):用户定义,可以修改,在当前进程使用,比如var=123等
与其他语言的区别:非类型性质,也就是不必指定变量是数字或字符串等。
3.关于环境变量:环境变量只能从父进程到子进程单向继承。换句话说:在子进程中的环境如何变更,均不会影响父进程的环境。
4.Shell脚本:其实就是将你平时在Shell prompt后所输入的多行command依序写入一个文件去而已
fork,source和exec方式执行Shell脚本
fork方式
也就是常用的方式,一般在shell直接输入脚本文件路径就可以了。这种方式由当前进程创建一个子进程
./mytest.sh
source方式
使用方式如下(source与”.”等价):
source ./mytest.sh
#或者
../mytest.sh
source方式的特点是,在不另外创建子进程,而是在当前的的Shell环境中执行。
exec方式
exec./mytest.sh
此方式的特点是,不另外创建子进程,但是会终止当前的shell执行(其实我觉得这样理解可能更准确:使用exec会在当前的进程空间创建一个子线程,然后终止当前线程的执行,到了新建的线程执行完之后,其实两个线程都终止了,也就是这个当前shell进程也就终止了)
测试一
创建test1.sh,内容如下:
#!/bin/sh
cd~/bin
pwd
分别使用三种方式执行,source与exec方式都会将目录更改应用到当前环境,不同的是,exec方式执行完之后,shell环境就不能够使用了,会自动重启一个新的shell环境(进程)
测试二
一个更为详实的测试
脚本1.sh
#!/bin/sh
A=B
echo"PID for 1.sh before exec/source/fork:$$"
exportA
echo"1.sh: /$A is $A"
case$1in
exec)
echo"using exec..."
exec./2.sh;;
source)
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/sh
echo"PID for 2.sh: $$"
echo"2.sh get /$A=$A from 1.sh"
A=C
exportA
echo"2.sh: /$A is $A"
分别使用三种方式执行1.sh脚本,结果如下:
~$ ./1.sh fork
PID for 1.sh before exec/source/fork:531
1.sh: $A is B
using fork by default...
PID for 2.sh:532
2.sh get $A=B from 1.sh
2.sh: $A is C
PID for 1.sh after exec/source/fork:531
1.sh: $A is B
~$ ./1.sh source
PID for 1.sh before exec/source/fork:533
1.sh: $A is B
using source...
PID for 2.sh:533
2.sh get $A=B from 1.sh
2.sh: $A is C
PID for 1.sh after exec/source/fork:533
1.sh: $A is C
~$ ./1.sh exec
PID for 1.sh before exec/source/fork:537
1.sh: $A is B
using exec...
PID for 2.sh:537
2.sh get $A=B from 1.sh
2.sh: $A is C
注意:使用exec执行时1.sh中最后两句并没有被执行