到最后一節(jié)來寫“開篇”,確實有點古怪。不過,在第一篇(數(shù)值操作)的開頭實際上也算是一個小的開篇,那里提到整個系列的前提是需要有一定的 Shell 編程基礎(chǔ),因此,為了能夠讓沒有 Shell 編程基礎(chǔ)的讀者也可以閱讀這個系列,我到最后來重寫這個開篇。開篇主要介紹什么是 Shell,Shell 運行環(huán)境,Shell 基本語法和調(diào)試技巧。
首先讓我們從下圖看看 Shell 在整個操作系統(tǒng)中所處的位置吧,該圖的外圓描述了整個操作系統(tǒng)(比如 Debian/Ubuntu/Slackware
等),內(nèi)圓描述了操作系統(tǒng)的核心(比如 Linux Kernel
),而 Shell
和 GUI
一樣作為用戶和操作系統(tǒng)之間的接口。
GUI
提供了一種圖形化的用戶接口,使用起來非常簡便易學;而 Shell
則為用戶提供了一種命令行的接口,接收用戶的鍵盤輸入,并分析和執(zhí)行輸入字符串中的命令,然后給用戶返回執(zhí)行結(jié)果,使用起來可能會復雜一些,但是由于占用的資源少,而且在操作熟練以后可能會提高工作效率,而且具有批處理的功能,因此在某些應用場合還非常流行。
Shell
作為一種用戶接口,它實際上是一個能夠解釋和分析用戶鍵盤輸入,執(zhí)行輸入中的命令,然后返回結(jié)果的一個解釋程序(Interpreter,例如在 linux
下比較常用的 Bash
),我們可以通過下面的命令查看當前的 Shell
:
$ echo $Shell
/bin/bash
$ ls -l /bin/bash
-rwxr-xr-x 1 root root 702160 2008-05-13 02:33 /bin/bash
該解釋程序不僅能夠解釋簡單的命令,而且可以解釋一個具有特定語法結(jié)構(gòu)的文件,這種文件被稱作腳本(Script)。它具體是如何解釋這些命令和腳本文件的,這里不深入分析,請看我在 2008 年寫的另外一篇文章:《Linux命令行上程序執(zhí)行的一剎那》。
既然該程序可以解釋具有一定語法結(jié)構(gòu)的文件,那么我們就可以遵循某一語法來編寫它,它有什么樣的語法,如何運行,如何調(diào)試呢?下面我們以 Bash
為例來討論這幾個方面。
為了方便后面的練習,我們先搭建一個基本運行環(huán)境:在一個 Linux 操作系統(tǒng)中,有一個運行有 Bash
的命令行在等待我們鍵入命令,這個命令行可以是圖形界面下的 Terminal
(例如 Ubuntu
下非常厲害的 Terminator
),也可以是字符界面的 Console
(可以用 CTRL+ALT+F1~6
切換),如果你發(fā)現(xiàn)當前 Shell
不是 Bash
,請用下面的方法替換它:
$ chsh $USER -s /bin/bash
$ su $USER
或者是簡單地鍵入Bash:
$ bash
$ echo $Shell # 確認一下
/bin/bash
如果沒有安裝 Linux 操作系統(tǒng),也可以考慮使用一些公共社區(qū)提供的 Linux 虛擬實驗服務,一般都有提供遠程 Shell
,你可以通過 Telnet
或者是 Ssh
的客戶端登錄上去進行練習。
有了基本的運行環(huán)境,那么如何來運行用戶鍵入的命令或者是用戶編寫好的腳本文件呢 ?
假設(shè)我們編寫好了一個 Shell 腳本,叫 test.sh
。
第一種方法是確保我們執(zhí)行的命令具有可執(zhí)行權(quán)限,然后直接鍵入該命令執(zhí)行它:
$ chmod +x /path/to/test.sh
$ /path/to/test.sh
第二種方法是直接把腳本作為 Bash
解釋器的參數(shù)傳入:
$ bash /path/to/test.sh
或
$ source /path/to/test.sh
或
$ . /path/to/test.sh
先來一個 Hello, World
程序。
下面來介紹一個 Shell 程序的基本結(jié)構(gòu),以 Hello, World
為例:
#!/bin/bash -v
# test.sh
echo "Hello, World"
把上述代碼保存為 test.sh
,然后通過上面兩種不同方式運行,可以看到如下效果。
方法一:
$ chmod +x test.sh
$ ./test.sh
./test.sh
#!/bin/bash -v
echo "Hello, World"
Hello, World
方法二:
$ bash test.sh
Hello, World
$ source test.sh
Hello, World
$ . test.sh
Hello, World
我們發(fā)現(xiàn)兩者運行結(jié)果有區(qū)別,為什么呢?這里我們需要關(guān)注一下 test.sh
文件的內(nèi)容,它僅僅有兩行,第二行打印了 Hello, World
,兩種方法都達到了目的,但是第一種方法卻多打印了腳本文件本身的內(nèi)容,為什么呢?
原因在該文件的第一行,當我們直接運行該腳本文件時,該行告訴操作系統(tǒng)使用用#!
符號之后面的解釋器以及相應的參數(shù)來解釋該腳本文件,通過分析第一行,我們發(fā)現(xiàn)對應的解釋器以及參數(shù)是 /bin/bash -v
,而 -v
剛好就是要打印程序的源代碼;但是我們在用第二種方法時沒有給 Bash
傳遞任何額外的參數(shù),因此,它僅僅解釋了腳本文件本身。
其他語法細節(jié)請直接看《Shell編程學習筆記》即本書后面的附錄一。
Shell 語言作為解釋型語言,它的程序設(shè)計過程跟編譯型語言有些區(qū)別,其基本過程如下:
可見它沒有編譯型語言的"麻煩的"編譯和鏈接過程,不過正是因為這樣,它出錯時調(diào)試起來不是很方便,因為語法錯誤和邏輯錯誤都在運行時出現(xiàn)。下面我們簡單介紹一下調(diào)試方法。
可以直接參考資料:Shell 腳本調(diào)試技術(shù) 或者 BASH 的調(diào)試手段。
Shell 語言作為一門解釋型語言,可以使用大量的現(xiàn)有工具,包括數(shù)值計算、符號處理、文件操作、網(wǎng)絡(luò)操作等,因此,編寫過程可能更加高效,但是因為它是解釋型的,需要在執(zhí)行過程中從磁盤上不斷調(diào)用外部的程序并進行進程之間的切換,在運行效率方面可能有劣勢,所以我們應該根據(jù)應用場合選擇使用 Shell 或是用其他的語言來編程。
更多建議: