shell環境
每個shell程式有一個自己的執行環境,不同的Shell程式有不同的Shell環境。Shell解析命令列、呼叫命令列的過程都在這個環境中完成。
呼叫shell程式時,會讀取配置檔案來初始化Shell環境。
讀取配置檔案情況分為兩種:
- 使用者登入啟動的shell
- 非使用者登入啟動的shell
什麼是子shell
所謂子shell,即從當前shell環境中新開了一個shell環境,這個新開的shell環境就是子shell,而開啟子shell的環境稱為該子shell的父shell。
子Shell的本質可以理解為Shell的子程式,子程式的概念是由父程式的概念引申而來的,在Linux系統中,系統執行的應用程式幾乎都是從init
(pid為1的程式)程式派生而來的,所有這些應用程式都可以視為init程式的子程式,而init
則為它們的父程式。通過執行pstree -a
命令就可以看到init
及系統中其他程式的程式樹資訊:
[root@test ~]# pstree -a
systemd --switched-root --system --deserialize 22
├─NetworkManager --no-daemon
│ └─2*[{NetworkManager}]
├─VGAuthService -s
├─agetty --noclear tty1 linux
├─auditd
│ └─{auditd}
├─chronyd
├─crond -n
├─dbus-daemon --system --address=systemd: --nofork --nopidfile --systemd-activation
├─irqbalance --foreground
├─lvmetad -f
├─master -w
│ ├─pickup -l -t unix -u
│ └─qmgr -l -t unix -u
├─polkitd --no-debug
│ └─6*[{polkitd}]
├─rsyslogd -n
│ └─2*[{rsyslogd}]
├─sshd -D
│ └─sshd
│ └─bash
│ └─pstree -a
├─systemd-journal
├─systemd-logind
├─systemd-udevd
├─tuned -Es /usr/sbin/tuned -l -P
│ └─4*[{tuned}]
└─vmtoolsd
└─2*[{vmtoolsd}]
Tips:若無pstree命令,請執行
yum -y install psmisc
安裝。
對於Shell的子程式來說,它是一個從父級Shell程式派生而來的新的Shell程式,我們將這種新的Shell程式稱為這個父級Shell的子Shell。
Shell指令碼是從上至下、從左至右依次執行每一行的命令及語句的,即執行完一個命令之後再執行下一個。如果在Shell指令碼中遇到子指令碼(即指令碼巢狀),就會先執行子指令碼的內容,完成後再返回父指令碼繼續執行父指令碼內後續的命令及語句。
子shell會從父shell中繼承很多環境,如變數、命令全路徑、檔案描述符、當前工作目錄、陷阱等等,但子shell有很多種型別,不同型別的子shell繼承的環境不相同。可以使用$BASH_SUBSHELL
變數來檢視從當前程式開始的子shell層數,$BASHPID
檢視當前所處BASH的PID,這不同於特殊變數$$
值,因為$$
在大多數情況下都會從父shell中繼承。
注意:子 Shell 雖然能使用父 Shell 的的一切,但是如果子 Shell 對資料做了修改,比如修改了全域性變數,那麼這種修改只能停留在子 Shell,無法傳遞給父 Shell。不管是子程式還是子 Shell,都是“傳子不傳父”。
子shell的分類
大致分為兩類:
- sub shell:通過程式替換
<(cmd),>(cmd)
、命令替換$(cmd)
、(cmd)
、|
或者$
隱式生成的子shell。因為父shell是通過fork
建立sub shell,因此子shell會從父shell中繼承很多環境,如變數、命令全路徑、檔案描述符、當前工作目錄、陷阱等等; - child shell:通過以可執行檔案的方式執行shell指令碼或直接在當前shell中啟動shell直譯器的方式得到的子shell。父shell通過
fork-exec
的方式建立子shell,導致父shell和子shell除了維持“父子關係”外,沒有其他關聯。
註釋:使用 fork() 函式可以建立一個子程式;除了 PID(程式ID)等極少的引數不同外,子程式的一切都來自父程式,包括程式碼、資料、堆疊、開啟的檔案等,就連程式碼的執行位置(狀態)都是一樣的。