這篇文章給大家介紹怎么理解Linux shell的知識(shí)點(diǎn),內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。
目前創(chuàng)新互聯(lián)已為數(shù)千家的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)頁(yè)空間、網(wǎng)站運(yùn)營(yíng)、企業(yè)網(wǎng)站設(shè)計(jì)、鐵西網(wǎng)站維護(hù)等服務(wù),公司將堅(jiān)持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長(zhǎng),共同發(fā)展。我個(gè)人很喜歡使用 Linux 系統(tǒng),雖然說(shuō) Windows 的圖形化界面做的確實(shí)比 Linux 好,但是對(duì) 腳本的支持太差了。一開(kāi)始有點(diǎn)不習(xí)慣 命令行操作,但是熟悉了之后反而發(fā)現(xiàn)移動(dòng)鼠標(biāo)點(diǎn)點(diǎn)點(diǎn)才是浪費(fèi)時(shí)間的罪魁禍?zhǔn)住D敲磳?duì)于 Linux 命令行,本文不是介紹某些命令的用法,而是說(shuō)明一些簡(jiǎn)單卻特別容易讓人迷惑的細(xì)節(jié)問(wèn)題。 |
一、標(biāo)準(zhǔn)輸入和參數(shù)的區(qū)別
這個(gè)問(wèn)題一定是最容易讓人迷惑的,具體來(lái)說(shuō),就是搞不清什么時(shí)候用管道符|和文件重定向>,<,什么時(shí)候用變量$。
比如說(shuō),我現(xiàn)在有個(gè)自動(dòng)連接寬帶的 shell 腳本connect.sh,存在我的家目錄:
$ where connect.sh /home/fdl/bin/connect.sh
如果我想刪除這個(gè)腳本,而且想少敲幾次鍵盤,應(yīng)該怎么操作呢?我曾經(jīng)這樣嘗試過(guò):
$ 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ù),前文畫(huà)圖解釋過(guò):
輸入重定向就是說(shuō),程序想讀取數(shù)據(jù)的時(shí)候就會(huì)去 files[0] 讀取,所以我們只要把 files[0] 指向一個(gè)文件,那么程序就會(huì)從這個(gè)文件中讀取數(shù)據(jù),而不是從鍵盤。
同理,輸出重定向就是把files[1]指向一個(gè)文件,那么程序的輸出就不會(huì)寫(xiě)入到顯示器,而是寫(xiě)入到這個(gè)文件中。
管道符其實(shí)也是異曲同工,把一個(gè)進(jìn)程的輸出流和另一個(gè)進(jìn)程的輸入流接起一條「管道」,數(shù)據(jù)就在其中傳遞:
Linux 進(jìn)程、線程、文件描述符的底層原理
用剛才的例子說(shuō),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
如果命令能夠讓終端阻塞,說(shuō)明該命令接收標(biāo)準(zhǔn)輸入,反之就是不接受,比如你只運(yùn)行cat命令不加任何參數(shù),終端就會(huì)阻塞,等待你輸入字符串并回顯相同的字符串。
二、后臺(tái)運(yùn)行程序
比如說(shuō)你遠(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)在你可以通過(guò)服務(wù)器的 IP 地址測(cè)試 Django 服務(wù),但是終端此時(shí)就阻塞了,你輸入什么都不響應(yīng),除非輸入 Ctrl-C 或者 Ctrl-/ 終止 python 進(jìn)程。
可以在命令之后加一個(gè)&符號(hào),這樣命令行不會(huì)阻塞,可以響應(yīng)你后續(xù)輸入的命令,但是如果你退出服務(wù)器的登錄,就不能訪問(wèn)該網(wǎng)頁(yè)了。
如果你想在退出服務(wù)器之后仍然能夠訪問(wèn) 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)程分出來(lái)的子進(jìn)程。正常情況下,shell 進(jìn)程會(huì)阻塞,等待子進(jìn)程退出才重新接收你輸入的新的命令。加上&號(hào),只是讓 shell 進(jìn)程不再阻塞,可以繼續(xù)響應(yīng)你的新命令。但是無(wú)論如何,你如果關(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命令就完全沒(méi)有影響了。
類似的,還有一種后臺(tái)運(yùn)行常用的做法是這樣:
$ nohup some_cmd &
nohup命令也是類似的原理,不過(guò)通過(guò)我的測(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ì)開(kāi)啟 shell 的命令回顯,你可以通過(guò)回顯觀察 shell 到底在執(zhí)行什么命令:
可見(jiàn) echo $(cmd) 和 echo "$(cmd)",結(jié)果差不多,但是仍然有區(qū)別。注意觀察,雙引號(hào)轉(zhuǎn)義完成的結(jié)果會(huì)自動(dòng)增加單引號(hào),而前者不會(huì)。
也就是說(shuō),如果 $ 讀取出的參數(shù)字符串包含空格,應(yīng)該用雙引號(hào)括起來(lái),否則就會(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)然是找不到的。
解決方法是使用腳本文件的路徑,而不是僅僅通過(guò)腳本名稱:
$ sudo /home/fdl/bin/connect.sh
關(guān)于怎么理解Linux shell的知識(shí)點(diǎn)就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。