用source,dot(.)的方式執行shell指令碼的時候,不產生子進程,shell指令碼在當前的shell中運行,shell指令碼運行完成後,在shell指令碼中聲明的變數在當前的shell中是可見的.
直接用指令檔名的方式執行shell指令碼的時候,產生子進程,shell指令碼在子進程中運行,shell指令碼運行完成後,在shell指令碼中聲明的變數在當前的shell中是不可見的.
驗證過程:
在目前的目錄下有一個tt.sh的指令碼內容如下:
echo $$
ttvar=12345
1,先來看當前的shell的pid:28210
test@btdalvm03:~/c$ echo $$
28210
2,以source的方式執行tt.sh,指令碼列印的pid和當前shell的pid一致,在tt.sh中定義的變數ttvar在指令碼執行完成後仍然可以訪問.
test@btdalvm03:~/c$ source tt.sh
28210
test@btdalvm03:~/c$ echo $ttvar
12345
3,以dot方式執行和source效果一樣,先用unset將ttvar變數清除.
test@btdalvm03:~/c$ unset ttvar
test@btdalvm03:~/c$ echo $ttvar
test@btdalvm03:~/c$ . tt.sh
28210
test@btdalvm03:~/c$ echo $ttvar
12345
4以指令檔名稱直接運行,要件當前檔案夾加入PATH,(或者以./tt.sh指定檔案名稱)
test@btdalvm03:~/c$ PATH=$PATH:.
test@btdalvm03:~/c$ unset ttvar
test@btdalvm03:~/c$ echo $ttvar
test@btdalvm03:~/c$ tt.sh
28796
test@btdalvm03:~/c$ echo $ttvar
test@btdalvm03:~/c$
可以看到這種方式,產生了新的子進程,指令碼運行完成後,裡面定義的變數對於當前的shell是不可訪問的.
在改變sh的時候也是要產生子進程的,通過exit退回到改變之前的sh.
test@btdalvm03:~/c$ echo $$28210test@btdalvm03:~/c$ echo $$28210test@btdalvm03:~/c$ shsh-3.2$ echo $$29152sh-3.2$ bashbash interactive changedtest@btdalvm03:~/c$ echo $$29153test@btdalvm03:~/c$ ps PID TTY TIME CMD28210 pts/1 00:00:00 bash29152 pts/1 00:00:00 sh29153 pts/1 00:00:00 bash29205 pts/1 00:00:00 pstest@btdalvm03:~/c$ exitexitsh-3.2$ echo $$29152sh-3.2$ exitexittest@btdalvm03:~/c$ echo $$28210test@btdalvm03:~/c$