從 Bash 和 Korn 到 C shell:評(píng)估 Linux 中的 shell
Shell 就像編輯器一樣:每個(gè)人都有自己喜歡的選擇并極力為該選擇辯護(hù)(還告訴您為什么應(yīng)該使用該選擇)。確實(shí)如此,shell 可提供不同的功能,但它們都實(shí)現(xiàn)了數(shù)十年前開(kāi)發(fā)的核心理念。
我第一次使用現(xiàn)代 shell 是在二十世紀(jì) 80 年代,當(dāng)時(shí)我正在 SunOS 上開(kāi)發(fā)軟件。當(dāng)我了解了將一個(gè)程序的輸出用作另一個(gè)程序的輸入(甚至多次連環(huán)地使用)的能力后,我就有了一種簡(jiǎn)單且高效的方式來(lái)創(chuàng)建過(guò)濾器和轉(zhuǎn)換。該核心理念提供了一種方式來(lái)構(gòu)建一些簡(jiǎn)單工具,這些工具足夠靈活,可與其他工具組合使用。通過(guò)這種方式,shell 不僅提供了一種與內(nèi)核和設(shè)備交互的方式,還提供了現(xiàn)在作為軟件開(kāi)發(fā)中的常見(jiàn)設(shè)計(jì)模式的集成服務(wù)(比如管道和過(guò)濾器)。
讓我們首先簡(jiǎn)單介紹一下現(xiàn)代 shell 的發(fā)展歷史,然后探討如今一些可用于 Linux 的外來(lái)的有用 shell。
shell 的發(fā)展歷史
Shell(或命令行解釋器)具有很長(zhǎng)的歷史了,但我們的討論從第一個(gè) UNIX® shell 開(kāi)始。(貝爾實(shí)驗(yàn)室的)Ken Thompson 于 1971 年開(kāi)發(fā)了第一個(gè)用于 UNIX 的 shell,名為 V6 shell。類似于它在 Multics 中的前身,這個(gè) shell (/bin/sh) 是一個(gè)在內(nèi)核外部執(zhí)行的獨(dú)立用戶程序。globbing(參數(shù)擴(kuò)展的模式匹配,比如 *.txt)等概念是在一個(gè)名為 glob的獨(dú)立實(shí)用程序中實(shí)現(xiàn)的,就像用于評(píng)估條件表達(dá)式的 if 命令一樣。這種獨(dú)立性可保持 shell 很小,只需不到 900 行 C 源代碼(請(qǐng)參見(jiàn) 參考資料 獲取原始源代碼的鏈接)。
該 shell 為重定向(< > 和 >>)和管道(| 或 ^)引入了一種緊湊的語(yǔ)法,這種語(yǔ)法已延續(xù)到現(xiàn)代 shell 中。您也可以找到對(duì)調(diào)用順序命令(使用 ;)和異步命令(使用 &)的支持。
Thompson shell 缺少的是編寫(xiě)腳本的能力。它的唯一用途就是用作一個(gè)交互式 shell(命令解釋器)來(lái)調(diào)用命令和查看結(jié)果。
1977 年以來(lái)的 UNIX shell
撇開(kāi) Thompson shell,我們開(kāi)始將目光轉(zhuǎn)移到 1977 年引入 Bourne shell 時(shí)的現(xiàn)代 shell。Bourne shell 由 Stephen Bourne 在 AT&T Bell Labs 為 V7 UNIX 創(chuàng)建,它在如今仍然是一個(gè)有用的 shell(在一些情況下還被用作默認(rèn)的根 shell)。作者在研究 ALGOL68 編譯器之后開(kāi)發(fā)了 Bourne shell,所以您會(huì)發(fā)現(xiàn)它的語(yǔ)法比其他 shell 更加類似于 Algorithmic Language (ALGOL)。盡管使用 C 開(kāi)發(fā),源代碼本身甚至使用了宏賦予它一種 ALGOL68 特色。
Bourne shell 有兩個(gè)主要目標(biāo):用作一個(gè)命令解釋器來(lái)交互式執(zhí)行操作系統(tǒng)命令,以及用于編寫(xiě)腳本(編寫(xiě)可通過(guò) shell 調(diào)用的可重用腳本)。 除了取代 Thompson shell,Bourne shell 還提供了相對(duì)于其前身的多項(xiàng)優(yōu)勢(shì)。Bourne 向腳本中引入了控制流、循環(huán)和變量,提供了一種更加強(qiáng)大的語(yǔ)言來(lái)與操作系統(tǒng)交互(包括交互式和非交互式)。該 shell 還允許您使用 shell 腳本作為過(guò)濾器,提供對(duì)處理信號(hào)的集成支持,但缺乏定義函數(shù)的能力。 最后,它整合了我們?nèi)缃袷褂玫脑S多功能,包括命令替換(使用反引號(hào))和用于將保留的字符串文字嵌入到腳本中的 HERE 文檔。
Bourne shell 不僅是向前發(fā)展的重要一步,也是眾多衍生的 shell 的基礎(chǔ),其中許多 shell 如今應(yīng)用在典型的 Linux 系統(tǒng)中。圖 1 演示了重要 shell 的系列。Bourne shell 導(dǎo)致了 Korn shell (ksh)、Almquist shell (ash) 和流行的 Bourne Again Shell(或 Bash)的開(kāi)發(fā)。在 Bourne shell 發(fā)布時(shí),C shell (csh) 正在開(kāi)發(fā)。圖 1 顯示了主要系列,但沒(méi)有展示所有影響,也沒(méi)有展示一些具有重要貢獻(xiàn)的 shell。
圖 1. 1977 年以來(lái)的 Linux shell
我們稍后將分析其中一些 shell,查看為它們的進(jìn)步做出貢獻(xiàn)的語(yǔ)言和功能示例。
一種假想的 shell 的基本架構(gòu)很簡(jiǎn)單(Bourne 的 shell 就是一個(gè)證據(jù))。在圖 2 中可以看到,基本架構(gòu)看起來(lái)類似一個(gè)管道,其中會(huì)分析和解析輸入,展開(kāi)符號(hào)(使用各種方法,比如括號(hào)、波浪號(hào)、變量、參數(shù)擴(kuò)展和替換,以及文件名生成),最終執(zhí)行命令(使用 shell 內(nèi)置的命令或外部命令)。
圖 2. 假想 shell 的簡(jiǎn)單架構(gòu)
在 參考資料 部分中,您可以找到一些鏈接來(lái)了解開(kāi)源 Bash shell 的架構(gòu)。
探索 Linux shell
現(xiàn)在讓我們看看其中一些 shell,回顧它們所做的貢獻(xiàn)并在每個(gè) shell 中查看示例腳本。查看的內(nèi)容包括 C shell、Korn shell 和 Bash。
Tenex C shell
C shell 是 Bill Joy 1978 年在加州大學(xué)伯克利分校攻讀研究生期間為 Berkeley Software Distribution (BSD) UNIX 系統(tǒng)開(kāi)發(fā)的。5 年后,該 shell 引入了來(lái)自 Tenex 系統(tǒng)(在 DEC PDP 系統(tǒng)上很流行)的功能。Tenex 引入了文件名和命令完成功能,以及命令行編輯功能。Tenex C shell (tcsh) 仍然向后兼容 csh,但改進(jìn)了它的整體交互式功能。tcsh 是 Ken Greer 在 Carnegie Mellon University 開(kāi)發(fā)的。
該 C shell 的一個(gè)重要的設(shè)計(jì)目標(biāo)是創(chuàng)建一種類似 C 語(yǔ)言的腳本語(yǔ)言。這是一個(gè)有用的目標(biāo),因?yàn)?nbsp;C 語(yǔ)言是所使用的主要語(yǔ)言(此外,該操作系統(tǒng)也是主要使用 C 語(yǔ)言開(kāi)發(fā)的)。
Bill Joy 在 C shell 中引用的一項(xiàng)有用功能是命令歷史。此功能維護(hù)以前執(zhí)行的命令的歷史,允許用戶檢查并輕松選擇之前的命令來(lái)執(zhí)行。例如,鍵入命令 history 將顯示以前執(zhí)行的命令。可使用向上和向下箭頭來(lái)選擇命令,或者可以使用 !! 執(zhí)行前一個(gè)命令。也可以引用以前的命令的參數(shù),例如 !* 引用前一個(gè)命令的所有參數(shù),其中 !$ 引用前一個(gè)命令的最后一個(gè)參數(shù)。
看一下 tcsh 腳本的一個(gè)簡(jiǎn)單示例(清單 1)。這段腳本獲取一個(gè)參數(shù)(一個(gè)目錄名稱),輸出該目錄中的所有可執(zhí)行文件以及找到的文件數(shù)量。我將在每個(gè)示例中重用此腳本設(shè)計(jì)來(lái)演示區(qū)別。
tcsh 腳本可分解為 3 個(gè)基本部分。首先,請(qǐng)注意,我使用了 shebang 或 hashbang 符號(hào)來(lái)將此文件聲明為可由定義的 shell 可執(zhí)行文件(在本例中為 tcsh 二進(jìn)制文件)解釋。這允許我以常規(guī)可執(zhí)行文件的形式執(zhí)行該文件,而不在它之前添加解釋器二進(jìn)制文件。它維護(hù)找到的可執(zhí)行文件數(shù)量,所以我將此數(shù)量初始化為 0。
清單 1. 用 tcsh 編寫(xiě)的查找所有可執(zhí)行文件的腳本
#!/bin/tcsh
# find all executables
set count=0
# Test arguments
if ($#argv != 1) then
echo "Usage is $0 <dir>"
exit 1
endif
# Ensure argument is a directory
if (! -d $1) then
echo "$1 is not a directory."
exit 1
endif
# Iterate the directory, emit executable files
foreach filename ($1/*)
if (-x $filename) then
echo $filename
@ count = $count + 1
endif
end
echo
echo "$count executable files found."
exit 0
|
第一部分測(cè)試用戶傳遞的參數(shù)。#argv 變量表示傳入的參數(shù)數(shù)量(不包括命令名稱本身)。您可指定這些參數(shù)的索引來(lái)訪問(wèn)它們:例如,#1 表示第一個(gè)參數(shù)(它是 argv[1] 的簡(jiǎn)寫(xiě))。該腳本需要一個(gè)參數(shù);如果它未找到該參數(shù),則輸出一條錯(cuò)誤消息,使用 $0 表示在控制臺(tái)輸入的命令名稱(argv[0])。
第二部分確保傳入的參數(shù)是一個(gè)目錄。如果該參數(shù)是一個(gè)目錄,-d 操作符返回 True。但請(qǐng)注意,我首先指定了一個(gè) ! 符號(hào),這表示無(wú)效。這樣,表達(dá)式可表明,如果參數(shù)不是一個(gè)目錄,則輸出一條錯(cuò)誤消息。
最后一部分迭代目錄中的文件,以測(cè)試它們是否可執(zhí)行文件。我使用方便的 foreach 迭代器,它循環(huán)括號(hào)(在本例中為該目錄)中的每一項(xiàng),然后在循環(huán)中測(cè)試每一項(xiàng)。這一步使用 -x 操作符測(cè)試文件是否為可執(zhí)行文件,如果是,則輸出該文件并將數(shù)量加一。在腳本的末尾,我輸出可執(zhí)行文件的數(shù)量。
Korn shell
Korn shell (ksh) 由 David Korn 設(shè)計(jì),是在與 Tenex C shell 相同的時(shí)期引入的。Korn shell 的一項(xiàng)最有趣的功能是,它除了向后兼容原始的 Bourne shell,還可用作腳本語(yǔ)言。
Korn shell 在 2000 年以開(kāi)源形式發(fā)布(依據(jù) Common Public License)以前一直是專用的軟件。除了提供與 Bourne shell 強(qiáng)大的向后兼容性,Korn shell 還包含其他 shell 的功能(比如 csh 的歷史功能)。該 shell 還提供了可在現(xiàn)代腳本語(yǔ)言(比如 Ruby 和 Python)中找到的一些更加高級(jí)的功能 — 例如,關(guān)聯(lián)數(shù)組和浮點(diǎn)算法。Korn shell 可用于多種操作系統(tǒng),包括 IBM® AIX® 和 HP-UX,致力于支持 Portable Operating System Interface for UNIX (POSIX) shell 語(yǔ)言標(biāo)準(zhǔn)。
Korn shell 是 Bourne shell 的衍生物,因此看上去更像 Bourne shell 和 Bash 而不是 C shell。讓我們看一個(gè)查找可執(zhí)行文件的 Korn shell 示例(清單 2)。
清單 2. 用 ksh 編寫(xiě)的查找所有可執(zhí)行文件的腳本
#!/usr/bin/ksh
# find all executables
count=0
# Test arguments
if [ $# -ne 1 ] ; then
echo "Usage is $0 <dir>"
exit 1
fi
# Ensure argument is a directory
if [ ! -d "$1" ] ; then
echo "$1 is not a directory."
exit 1
fi
# Iterate the directory, emit executable files
for filename in "$1"/*
do
if [ -x "$filename" ] ; then
echo $filename
count=$((count+1))
fi
done
echo
echo "$count executable files found."
exit 0
|
在清單 2 中,您將注意到的第一點(diǎn)是它與 清單 1 的相似性。在結(jié)構(gòu)上,該腳本基本上是相同的,但在執(zhí)行條件、表達(dá)式和迭代的方式上存在明顯的區(qū)別。沒(méi)有采用類似 C 的測(cè)試操作符,ksh 采用了典型的 Bourne 風(fēng)格操作符(-eq、-ne 和 -lt 等)。
Korn shell 也有用一些與迭代相關(guān)的區(qū)別。在 Korn shell 中,使用了 for in 結(jié)構(gòu),使用命令替換來(lái)表示從命令 ls '$1/*(表示指定子目錄的內(nèi)容)的標(biāo)準(zhǔn)輸出創(chuàng)建的文件列表。
除了上面定義的其他功能,Korn 還支持別名功能(用于將一個(gè)詞替換為用戶定義的字符串)。Korn 還有其他許多功能默認(rèn)已禁用(比如文件名稱完成),但可由用戶啟用。
Bourne-Again Shell(或 Bash)是一個(gè)開(kāi)源 GNU 項(xiàng)目,旨在取代 Bourne shell。Bash 由 Brian Fox 開(kāi)發(fā),已成為世上最流行的 shell 之一(出現(xiàn)在 Linux、Darwin、Windows®、Cygwin、Novell、Haiku 等系統(tǒng)中)。顧名思義,Bash 是 Bourne shell 的一個(gè)超集,大部分 Bourne 腳本都可原封不動(dòng)地執(zhí)行。
除了支持腳本的向后兼容性,Bash 還整合了來(lái)自 Korn 和 C shell 的功能。您將找到命令歷史、命令行編輯、一個(gè)目錄棧(pushd 和popd)、許多有用的環(huán)境變量和命令完成等。
Bash 繼續(xù)在發(fā)展,擁有許多新功能,支持正則表達(dá)式(類似于 Perl)和關(guān)聯(lián)數(shù)組。盡管其中一些功能可能在其他腳本語(yǔ)言中不存在,但可以編寫(xiě)兼容其他語(yǔ)言的腳本。對(duì)于這一點(diǎn),清單 3 中所示的示例腳本等同于 Korn shell 腳本(來(lái)自 清單 2),除了 shebang 區(qū)別 (/bin/bash)。
清單 3. 用 Bash 編寫(xiě)的查找所有可執(zhí)行文件的腳本
#!/bin/bash
# find all executables
count=0
# Test arguments
if [ $# -ne 1 ] ; then
echo "Usage is $0 <dir>"
exit 1
fi
# Ensure argument is a directory
if [ ! -d "$1" ] ; then
echo "$1 is not a directory."
exit 1
fi
# Iterate the directory, emit executable files
for filename in "$1"/*
do
if [ -x "$filename" ] ; then
echo $filename
count=$((count+1))
fi
done
echo
echo "$count executable files found."
exit 0
|
這些 shell 之間的一個(gè)關(guān)鍵區(qū)別是它們的發(fā)布所依據(jù)的許可。您可能已猜到,Bash 是在 GNU 項(xiàng)目中開(kāi)發(fā)的,是依據(jù) GPL 發(fā)布的,而 csh、tcsh、zsh、ash 和 scsh 都是依據(jù) BSD 或一種類似 BSD 的許可來(lái)發(fā)布的。Korn shell 可依據(jù) Common Public License 使用。
外來(lái)的 shell
對(duì)于大膽的開(kāi)發(fā)人員,可基于您的需要或愛(ài)好使用替代的 shell。Scheme shell (scsh) 提供了一種使用 Scheme(Lisp 語(yǔ)言的一種衍生物)的腳本環(huán)境。Pyshell 是對(duì)創(chuàng)建使用 Python 語(yǔ)言的類似腳本的一次嘗試。最后,對(duì)于嵌入式系統(tǒng),可以使用 BusyBox,它將一個(gè) shell 和所有命令合并到一個(gè)二進(jìn)制文件中,以簡(jiǎn)化其分發(fā)和管理。
清單 4 給出了以 Scheme shell (scsh) 編寫(xiě)的查找所有可執(zhí)行文件的腳本。這段腳本可能看起來(lái)很奇怪,但它實(shí)現(xiàn)了與我們目前所提供的腳本類似的功能。這段腳本包含 3 個(gè)函數(shù),直接使用可執(zhí)行代碼(在末尾)來(lái)測(cè)試參數(shù)數(shù)量。這段腳本真正強(qiáng)大之處在showfiles 函數(shù)內(nèi),它迭代一個(gè)列表(在 with-cwd 后構(gòu)造),在列表的每個(gè)元素后調(diào)用 write-ln。這個(gè)列表通過(guò)迭代指定的目錄并在其中過(guò)濾是可執(zhí)行文件的文件來(lái)生成。
清單 4. 用 scsh 編寫(xiě)的查找所有可執(zhí)行文件的腳本
#!/usr/bin/scsh -s
!#
(define argc
(length command-line-arguments))
(define (write-ln x)
(display x) (newline))
(define (showfiles dir)
(for-each write-ln
(with-cwd dir
(filter file-executable? (directory-files "." #t)))))
(if (not (= argc 1))
(write-ln "Usage is fae.scsh dir")
(showfiles (argv 1)))
|
結(jié)束語(yǔ)
早期 shell 的許多理念和大量接口在之后的 35 年幾乎未變 — 這是對(duì)早期 shell 的原始作者的貢獻(xiàn)的有力證明。在一個(gè)不斷自我改造的行業(yè)中,shell 已大大改進(jìn),但沒(méi)有發(fā)生重大變化。盡管存在過(guò)創(chuàng)建特殊 shell 的嘗試,但 Bourne shell 的衍生物仍然是所使用的主要 shell。
原文:http://www.ibm.com/developerworks/cn/linux/l-linux-shells/index.html?ca=drs-






















