這篇文章給大家介紹怎么理解Linux shell的知識(shí)點(diǎn),內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。
讓客戶滿意是我們工作的目標(biāo),不斷超越客戶的期望值來自于我們對(duì)這個(gè)行業(yè)的熱愛。我們立志把好的技術(shù)通過有效、簡(jiǎn)單的方式提供給客戶,將通過不懈努力成為客戶在信息化領(lǐng)域值得信任、有價(jià)值的長(zhǎng)期合作伙伴,公司提供的服務(wù)項(xiàng)目有:空間域名、網(wǎng)站空間、營(yíng)銷軟件、網(wǎng)站建設(shè)、欽南網(wǎng)站維護(hù)、網(wǎng)站推廣。
我個(gè)人很喜歡使用 Linux 系統(tǒng),雖然說 Windows 的圖形化界面做的確實(shí)比 Linux 好,但是對(duì) 腳本的支持太差了。一開始有點(diǎn)不習(xí)慣 命令行操作,但是熟悉了之后反而發(fā)現(xiàn)移動(dòng)鼠標(biāo)點(diǎn)點(diǎn)點(diǎn)才是浪費(fèi)時(shí)間的罪魁禍?zhǔn)?。那么?duì)于 Linux 命令行,本文不是介紹某些命令的用法,而是說明一些簡(jiǎn)單卻特別容易讓人迷惑的細(xì)節(jié)問題。 |
一、標(biāo)準(zhǔn)輸入和參數(shù)的區(qū)別
這個(gè)問題一定是最容易讓人迷惑的,具體來說,就是搞不清什么時(shí)候用管道符|和文件重定向>,<,什么時(shí)候用變量$。
比如說,我現(xiàn)在有個(gè)自動(dòng)連接寬帶的 shell 腳本connect.sh,存在我的家目錄:
$ where connect.sh /home/fdl/bin/connect.sh
如果我想刪除這個(gè)腳本,而且想少敲幾次鍵盤,應(yīng)該怎么操作呢?我曾經(jīng)這樣嘗試過:
$ where connect.sh | rm
實(shí)際上,這樣操作是錯(cuò)誤的,正確的做法應(yīng)該是這樣的:
$ rm $(where connect.sh)
前者試圖將where的結(jié)果連接到rm的標(biāo)準(zhǔn)輸入,后者試圖將結(jié)果作為命令行參數(shù)傳入。
標(biāo)準(zhǔn)輸入就是編程語(yǔ)言中諸如scanf或者readline這種命令;而參數(shù)是指程序的main函數(shù)傳入的args字符數(shù)組。
管道符和重定向符是將數(shù)據(jù)作為程序的標(biāo)準(zhǔn)輸入,而$(cmd)是讀取cmd命令輸出的數(shù)據(jù)作為參數(shù),前文畫圖解釋過:
輸入重定向就是說,程序想讀取數(shù)據(jù)的時(shí)候就會(huì)去 files[0] 讀取,所以我們只要把 files[0] 指向一個(gè)文件,那么程序就會(huì)從這個(gè)文件中讀取數(shù)據(jù),而不是從鍵盤。
同理,輸出重定向就是把files[1]指向一個(gè)文件,那么程序的輸出就不會(huì)寫入到顯示器,而是寫入到這個(gè)文件中。
管道符其實(shí)也是異曲同工,把一個(gè)進(jìn)程的輸出流和另一個(gè)進(jìn)程的輸入流接起一條「管道」,數(shù)據(jù)就在其中傳遞:
Linux 進(jìn)程、線程、文件描述符的底層原理
用剛才的例子說,rm命令源代碼中肯定不接受標(biāo)準(zhǔn)輸入,而是接收命令行參數(shù),刪除相應(yīng)的文件。作為對(duì)比,cat命令是既接受標(biāo)準(zhǔn)輸入,又接受命令行參數(shù):
$ cat filename ...file text... $ cat < filename ...file text... $ echo 'hello world' | cat hello world
如果命令能夠讓終端阻塞,說明該命令接收標(biāo)準(zhǔn)輸入,反之就是不接受,比如你只運(yùn)行cat命令不加任何參數(shù),終端就會(huì)阻塞,等待你輸入字符串并回顯相同的字符串。
二、后臺(tái)運(yùn)行程序
比如說你遠(yuǎn)程登錄到服務(wù)器上,運(yùn)行一個(gè) Django web 程序:
$ python manager.py runserver 0.0.0.0 Listening on 0.0.0.0:8080...
現(xiàn)在你可以通過服務(wù)器的 IP 地址測(cè)試 Django 服務(wù),但是終端此時(shí)就阻塞了,你輸入什么都不響應(yīng),除非輸入 Ctrl-C 或者 Ctrl-/ 終止 python 進(jìn)程。
可以在命令之后加一個(gè)&符號(hào),這樣命令行不會(huì)阻塞,可以響應(yīng)你后續(xù)輸入的命令,但是如果你退出服務(wù)器的登錄,就不能訪問該網(wǎng)頁(yè)了。
如果你想在退出服務(wù)器之后仍然能夠訪問 web 服務(wù),應(yīng)該這樣把命令包裹成這樣(cmd &):
$ (python manager.py runserver 0.0.0.0 &) Listening on 0.0.0.0:8080... $ logout
底層原理是這樣的:
每一個(gè)命令行終端都是一個(gè) shell 進(jìn)程,你在這個(gè)終端里執(zhí)行的程序?qū)嶋H上都是這個(gè) shell 進(jìn)程分出來的子進(jìn)程。正常情況下,shell 進(jìn)程會(huì)阻塞,等待子進(jìn)程退出才重新接收你輸入的新的命令。加上&號(hào),只是讓 shell 進(jìn)程不再阻塞,可以繼續(xù)響應(yīng)你的新命令。但是無論如何,你如果關(guān)掉了這個(gè) shell 命令行端口,依附于它的所有子進(jìn)程都會(huì)退出。
而(cmd &)這樣運(yùn)行命令,則是將cmd命令掛到一個(gè)systemd系統(tǒng)守護(hù)進(jìn)程名下,認(rèn)systemd做爸爸,這樣當(dāng)你退出當(dāng)前終端時(shí),對(duì)于剛才的cmd命令就完全沒有影響了。
類似的,還有一種后臺(tái)運(yùn)行常用的做法是這樣:
$ nohup some_cmd &
nohup命令也是類似的原理,不過通過我的測(cè)試,還是(cmd &)這種形式更加穩(wěn)定。
三、單引號(hào)和雙引號(hào)的區(qū)別
不同的 shell 行為會(huì)有細(xì)微區(qū)別,但有一點(diǎn)是確定的,對(duì)于$,(,)這幾個(gè)符號(hào),單引號(hào)包圍的字符串不會(huì)做任何轉(zhuǎn)義,雙引號(hào)包圍的字符串會(huì)轉(zhuǎn)義。
shell 的行為可以測(cè)試,使用set -x命令,會(huì)開啟 shell 的命令回顯,你可以通過回顯觀察 shell 到底在執(zhí)行什么命令:
可見 echo $(cmd) 和 echo "$(cmd)",結(jié)果差不多,但是仍然有區(qū)別。注意觀察,雙引號(hào)轉(zhuǎn)義完成的結(jié)果會(huì)自動(dòng)增加單引號(hào),而前者不會(huì)。
也就是說,如果 $ 讀取出的參數(shù)字符串包含空格,應(yīng)該用雙引號(hào)括起來,否則就會(huì)出錯(cuò)。
四、sudo 找不到命令
有時(shí)候我們普通用戶可以用的命令,用sudo加權(quán)限之后卻報(bào)錯(cuò) command not found:
$ connect.sh network-manager: Permission denied $ sudo connect.sh sudo: command not found
原因在于,connect.sh這個(gè)腳本僅存在于該用戶的環(huán)境變量中:
$ where connect.sh /home/fdl/bin/connect.sh
當(dāng)使用sudo時(shí),系統(tǒng)認(rèn)為是 root 用戶在執(zhí)行命令,所以會(huì)去搜索 root 用戶的環(huán)境變量,而這個(gè)腳本在 root 的環(huán)境變量目錄中當(dāng)然是找不到的。
解決方法是使用腳本文件的路徑,而不是僅僅通過腳本名稱:
$ sudo /home/fdl/bin/connect.sh
關(guān)于怎么理解Linux shell的知識(shí)點(diǎn)就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。