- 論壇徽章:
- 0
|
Bourne shell 有兩個(gè)主要目標(biāo):用作一個(gè)命令解釋器來(lái)交互式執(zhí)行操作系統(tǒng)命令,以及用于編寫腳本(編寫可通過 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 腳本作為過濾器,提供對(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)的開發(fā)。在 Bourne shell 發(fā)布時(shí),C shell (csh) 正在開發(fā)。圖 1 顯示了主要系列,但沒有展示所有影響,也沒有展示一些具有重要貢獻(xiàn)的 shell。
圖 1. 1977 年以來(lái)的 Linux shell
我們稍后將分析其中一些 shell,查看為它們的進(jìn)步做出貢獻(xiàn)的語(yǔ)言和功能示例。
基本 shell 架構(gòu)
一種假想的 shell 的基本架構(gòu)很簡(jiǎn)單(Bourne 的 shell 就是一個(gè)證據(jù))。在圖 2 中可以看到,基本架構(gòu)看起來(lái)類似一個(gè)管道,其中會(huì)分析和解析輸入,展開符號(hào)(使用各種方法,比如括號(hào)、波浪號(hào)、變量、參數(shù)擴(kuò)展和替換,以及文件名生成),最終執(zhí)行命令(使用 shell 內(nèi)置的命令或外部命令)。
圖 2. 假想 shell 的簡(jiǎn)單架構(gòu)
在 參考資料 部分中,您可以找到一些鏈接來(lá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 SoftwareDistribution (BSD) UNIX 系統(tǒng)開發(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 開發(fā)的。
該 C shell 的一個(gè)重要的設(shè)計(jì)目標(biāo)是創(chuàng)建一種類似 C 語(yǔ)言的腳本語(yǔ)言。這是一個(gè)有用的目標(biāo),因?yàn)?C 語(yǔ)言是所使用的主要語(yǔ)言(此外,該操作系統(tǒng)也是主要使用 C 語(yǔ)言開發(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 編寫的查找所有可執(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)訪問它們:例如,#1 表示第一個(gè)參數(shù)(它是 argv[1] 的簡(jiǎn)寫)。該腳本需要一個(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 年以開源形式發(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 編寫的查找所有可執(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ū)別。沒有采用類似 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
Bourne-Again Shell(或 Bash)是一個(gè)開源 GNU 項(xiàng)目,旨在取代 Bourne shell。Bash 由 Brian Fox 開發(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ǔ)言中不存在,但可以編寫兼容其他語(yǔ)言的腳本。對(duì)于這一點(diǎn),清單 3 中所示的示例腳本等同于 Korn shell 腳本(來(lái)自 清單 2),除了 shebang 區(qū)別 (/bin/bash)。
清單 3. 用 Bash 編寫的查找所有可執(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)目中開發(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ì)于大膽的開發(fā)人員,可基于您的需要或愛好使用替代的 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) 編寫的查找所有可執(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è)列表通過迭代指定的目錄并在其中過濾是可執(zhí)行文件的文件來(lái)生成。
清單 4. 用 scsh 編寫的查找所有可執(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),但沒有發(fā)生重大變化。盡管存在過創(chuàng)建特殊 shell 的嘗試,但 Bourne shell 的衍生物仍然是所使用的主要 shell。
|
|