shell的命令(轉)

post0發表於2007-08-11
shell的命令(轉)[@more@]

shell是使用者和Linux作業系統之間的介面。Linux中有多種shell,其中預設使用的是Bash。本章講述了shell的工作原理,shell的種類,shell的一般操作及Bash的特性。

 什麼是shell

Linux系統的shell作為作業系統的外殼,為使用者提供使用作業系統的介面。它是命令語言、命令解釋程式及程式設計語言的統稱。

shell是使用者和Linux核心之間的介面程式,如果把Linux核心想象成一個球體的中心,shell就是圍繞核心的外層。當從shell或其他程式向Linux傳遞命令時,核心會做出相應的反應。

shell是一個命令語言直譯器,它擁有自己內建的shell命令集,shell也能被系統中其他應用程式所呼叫。使用者在提示符下輸入的命令都由shell先解釋然後傳給Linux核心。

有一些命令,比如改變工作目錄命令cd,是包含在shell內部的。還有一些命令,例如複製命令cp和移動命令rm,是存在於檔案系統中某個目錄下的單獨的程式。對使用者而言,不必關心一個命令是建立在shell內部還是一個單獨的程式。

shell首先檢查命令是否是內部命令,若不是再檢查是否是一個應用程式(這裡的應用程式可以是Linux本身的實用程式,如ls和rm,也可以是購買的商業程式,如xv,或者是自由軟體,如emacs)。然後shell在搜尋路徑裡尋找這些應用程式(搜尋路徑就是一個能找到可執行程式的目錄列表)。如果鍵入的命令不是一個內部命令並且在路徑裡沒有找到這個可執行檔案,將會顯示一條錯誤資訊。如果能夠成功找到命令,該內部命令或應用程式將被分解為系統呼叫並傳給Linux核心。

 

shell的另一個重要特性是它自身就是一個解釋型的程式設計語言,shell程式設計語言支援絕大多數在高階語言中能見到的程式元素,如函式、變數、陣列和程式控制結構。shell程式語言簡單易學,任何在提示符中能鍵入的命令都能放到一個可執行的shell程式中。

當普通使用者成功登入,系統將執行一個稱為shell的程式。正是shell程式提供了命令列提示符。作為預設值(TurboLinux系統預設的shell是BASH),對普通使用者用“$”作提示符,對超級使用者(root)用“#”作提示符。

一旦出現了shell提示符,就可以鍵入命令名稱及命令所需要的引數。shell將執行這些命令。如果一條命令花費了很長的時間來執行,或者在螢幕上產生了大量的輸出,可以從鍵盤上按ctrl+c發出中斷訊號來中斷它(在正常結束之前,中止它的執行)。

當使用者準備結束登入對話程式時,可以鍵入logout命令、exit命令或檔案結束符(EOF)(按ctrl+d實現),結束登入。

我們來實習一下shell是如何工作的。

$ make work

make:***No rule to make target ‘work’. Stop.

$

註釋:make是系統中一個命令的名字,後面跟著命令引數。在接收到這個命令後,shell便執行它。本例中,由於輸入的命令引數不正確,系統返回資訊後停止該命令的執行。

在例子中,shell會尋找名為make的程式,並以work為引數執行它。make是一個經常被用來編譯大程式的程式,它以引數作為目標來進行編譯。在“make work”中,make編譯的目標是work。因為make找不到以work為名字的目標,它便給出錯誤資訊表示執行失敗,使用者又回到系統提示符下。

另外,使用者鍵入有關命令列後,如果shell找不到以其中的命令名為名字的程式,就會給出錯誤資訊。例如,如果使用者鍵入:

$ myprog

bash:myprog:command not found

$

可以看到,使用者得到了一個沒有找到該命令的錯誤資訊。使用者敲錯命令後,系統一般會給出這樣的錯誤資訊。

shell的種類

Linux中的shell有多種型別,其中最常用的幾種是Bourne shell(sh)、C shell(csh)和Korn shell(ksh)。三種shell各有優缺點。Bourne shell是UNIX最初使用的shell,並且在每種UNIX上都可以使用。Bourne shell在shell程式設計方面相當優秀,但在處理與使用者的互動方面做得不如其他幾種shell。Linux作業系統預設的shell是Bourne Again shell,它是Bourne shell的擴充套件,簡稱Bash,與Bourne shell完全向後相容,並且在Bourne shell的基礎上增加、增強了很多特性。Bash放在/bin/bash中,它有許多特色,可以提供如命令補全、命令編輯和命令歷史表等功能,它還包含了很多C shell和Korn shell中的優點,有靈活和強大的程式設計介面,同時又有很友好的使用者介面。

C shell是一種比Bourne shell更適於程式設計的shell,它的語法與C語言很相似。 Linux為喜歡使用C shell的人提供了Tcsh。Tcsh是C shell的一個擴充套件版本。Tcsh包括命令列編輯、可程式設計單詞補全、拼寫校正、歷史命令替換、作業控制和類似C語言的語法,它不僅和Bash shell是提示符相容,而且還提供比Bash shell更多的提示符引數。

Korn shell集合了C shell和Bourne shell的優點並且和Bourne shell完全相容。Linux系統提供了pdksh(ksh的擴充套件),它支援任務控制,可以在命令列上掛起、後臺執行、喚醒或終止程式。

Linux並沒有冷落其他shell使用者,還包括了一些流行的shell如ash、zsh等。每個shell都有它的用途,有些shell是有專利的,有些能從Internet網上或其他來源獲得。要決定使用哪個shell,只需讀一下各種shell的聯機幫助,並試用一下。

使用者在登入到Linux時由/etc/passwd檔案來決定要使用哪個shell。例如:

# fgrep lisa /etc/passwd

lisa:x:500:500:TurboLinux User:/home/lisa:/bin/bash

shell被列每行的末尾(/bin/bash)。

由於Bash是Linux上預設的shell,本章主要介紹Bash及其相關知識。

shell命令

命令列c

使用者登入到Linux系統時,可以看到一個shell提示符,標識了命令列的開始。使用者可以在提示符後面輸入任何命令及引數。例如:

$ date

二 11 23 01:34:58 CST 1999

$

使用者登入時,實際進入了shell,它遵循一定的語法將輸入的命令加以解釋並傳給系統。命令列中輸入的第一個字必須是一個命令的名字,第二個字是命令的選項或引數,命令列中的每個字必須由空格或TAB隔開,格式如下:

$ Command Option Arguments

1. 選項和引數

選項是包括一個或多個字母的程式碼,它前面有一個減號(減號是必要的,Linux用它來區別選項和引數),選項可用於改變命令執行的動作的型別。例如:

$ ls

motd passwd

$

這是沒有選項的ls命令,可列出當前目錄中所有檔案,只列出各個檔案的名字,而不顯示其他更多的資訊。

$ ls -l

total 2

-rw-r--r-- 2 wzh book 22 Apr 20 20:37 motd

-rw-r--r-- 2 wzh book 796 Apr 20 20:37 passwd

$

加入-l選項,將會為每個檔案列出一行資訊,諸如資料大小和資料最後被修改的時間。

大多數命令都被設計為可以接納引數。引數是在命令列中的選項之後鍵入的一個或多個單詞,例如:

$ ls -l text

-rw-r--r-- 2 wzh book 22 Apr 20 20:37 motd

-rw-r--r-- 2 wzh book 796 Apr 20 20:37 passwd

$

將顯示text目錄下的所有檔案及其資訊。

有些命令,如ls可以帶引數,而有一些命令可能需要一些最小數目的引數。例如,cp命令至少需要兩個引數,如果引數的數目與命令要求不符,shell將會給出出錯資訊。例如:

$ cp -i mydata newdata

注意:命令列中選項先於引數輸入。

2. 命令列特徵

命令列實際上是可以編輯的一個文字緩衝區,在按回車之前,可以對輸入的文字進行編輯。比如利用BACKSPACE鍵可以刪除剛鍵入的字元,可以進行整行刪除,還可以插入字元,使得使用者在輸入命令,尤其是複雜命令時,若出現鍵入錯誤,無須重新輸入整個命令,只要利用編輯操作,即可改正錯誤。

利用上箭頭可以重新顯示剛執行的命令,利用這一功能可以重複執行以前執行過的命令,而無須重新鍵入該命令。

bash儲存著以前鍵入過的命令的列表,這一列表被稱為命令歷史表。按動上箭頭,便可以在命令列上逐次顯示各條命令。同樣,按動下箭頭可以在命令列表中向下移動,這樣可以將以前的各條命令顯示在命令列上,使用者可以修改並執行這些命令。這一特徵將在10.4節中進行詳細的論述。

在一個命令列中還可以置入多個命令,用分號將各個命令隔開。例如:

$ ls -F;cp -i mydata newdata

也可以在幾個命令列中輸入一個命令,用反斜槓將一個命令列持續到下一行。

$ cp –i

mydata

newdata

 

上面的cp命令是在三行中輸入的,開始的兩行以反斜槓結束,把三行作為一個命令列。

shell中的特殊字元

shell中除使用普通字元外,還可以使用一些具有特殊含義和功能的特殊字元。在使用它們時應注意其特殊的含義和作用範圍。下面分別對這些特殊字元加以介紹。

1. 萬用字元

萬用字元用於模式匹配,如檔名匹配、路經名搜尋、字串查詢等。常用的萬用字元有*、?和括在方括號[ ]中的字元序列。使用者可以在作為命令引數的檔名中包含這些萬用字元,構成一個所謂的“模式串”,在執行過程中進行模式匹配。

* 代表任何字串(長度可以不等),例如:“f*”匹配以f打頭的任意字串。但應注意,檔名前的圓點(.)和路經名中的斜線(/)必須顯式匹配。例如“*”不能匹配.file,而“.*”才可以匹配.file。

? 代表任何單個字元。

[] 代表指定的一個字元範圍,只要檔名中[ ]位置處的字元在[ ]中指定的範圍之內,那麼這個檔名就與這個模式串匹配。方括號中的字元範圍可以由直接給出的字元組成,也可以由表示限定範圍的起始字元、終止字元及中間的連字元(-)組成。例如,f [a- d] 與f [abcd]的作用相同。Shell將把與命令列中指定的模式串相匹配的所有檔名都作為命令的引數,形成最終的命令,然後再執行這個命令。

下面我們給出表10-1說明這些萬用字元的具體含義。

表10-1 萬用字元含義舉例

模式串

意 義

*

當前目錄下所有檔案的名稱。

*Text*

當前目錄下所有檔名中包含有Text的檔案的名稱。

[ab-dm]*

當前目錄下所有以a、b、c、d、m開頭的檔案的名稱。

[ab-dm]?

當前目錄下所有以a、b、c、d、m開頭且後面只跟有一個字元的檔案的名稱。

/usr/bin/??

目錄/usr/bin下所有名稱為兩個字元的檔案的名稱。

 

特別需要注意的是,連字元“-”僅在方括號內有效,表示字元範圍,如在方括號外面就成為普通字元了。而*和?只在方括號外面是萬用字元,若出現在方括號之內,它們也失去萬用字元的能力,成為普通字元了。例如,模式“- a[*?]abc”中只有一對方括號是萬用字元,*和?均為普通字元,因此,它匹配的字串只能是- a*abc和- a?abc。

最後說明一下使用萬用字元時需要注意的一些問題。由於*、?和[]對於shell來說具有比較特殊的意義,因此在正常的檔名中不應出現這些字元。特別是在目錄名中不要出現它們,否則Shell匹配起來可能會無窮的遞迴下去。另外要注意的一點是:如果目錄中沒有與指定的模式串相匹配的檔名,那麼Shell將使用此模式串本身作為引數傳給有關命令。這可能就是命令中出現特殊字元的原因所在。

2. 引號

在shell中引號分為三種:單引號,雙引號和反引號。

* 單引號 ‘

由單引號括起來的字元都作為普通字元出現。特殊字元用單引號括起來以後,也會失去原有意義,而只作為普通字元解釋。例如:

$ string=’$PATH’

$ echo $string

$PATH

$

可見$保持了其本身的含義,作為普通字元出現。

* 雙引號 “

由雙引號括起來的字元,除$、、’、和”這幾個字元仍是特殊字元並保留其特殊功能外,其餘字元仍作為普通字元對待。對於$來說,就是用其後指定的變數的值來代替這個變數和$;對於而言,是跳脫字元,它告訴shell不要對其後面的那個字元進行特殊處理,只當作普通字元即可。可以想見,在雙引號中需要在前面加上的只有四個字元$,,’和”本身。而對”號,若其前面沒有加,則Shell會將它同前一個”號匹配。

例如,我們假定PATH的值為.:/usr/bin:/bin,輸入如下命令:

$ TestString=”$PATH”$PATH”

$ echo $TestString

.:/usr/bin:/ bin”$PATH

$

讀者可以自己試一下在第二個雙引號之前不加會產生什麼結果。

 

* 反引號 `

反引號(`)這個字元所對應的鍵一般位於鍵盤的左上角,不要將其同單引號(’)混淆。反引號括起來的字串被shell解釋為命令列,在執行時,shell首先執行該命令列,並以它的標準輸出結果取代整個反引號(包括兩個反引號)部分。例如:

$ pwd

/home/xyz

$ string=”current directory is `pwd`”

$ echo $string

current directour is /home/xyz

$

shell執行echo命令時,首先執行`pwd`中的命令pwd,並將輸出結果/home/xyz取代`pwd`這部分,最後輸出替換後的整個結果。

利用反引號的這種功能可以進行命令置換,即把反引號括起來的執行結果賦值給指定變數。例如:

$ today=`date`

$ echo Today is $today

Today is Mon Apr 15 16:20:13 CST 1999

$

反引號還可以巢狀使用。但需注意,巢狀使用時內層的反引號必須用反斜線()將其轉義。例如:

$ abc=`echo The number of users is `who| wc-l``

$ echo $abc

The number of users is 5

$

在反引號之間的命令列中也可以使用shell的特殊字元。Shell為得到``中命令的結果,它實際上要去執行``中指定的命令。執行時,命令中的特殊字元,如$,”,?等又將具有特殊含義,並且``所包含的可以是任何一個合法的Shell命令,如:

$ ls

note readme.txt Notice Unix.dir

$ TestString=”`echo $HOME ` ` ls [nN]*`”

$ echo $TestString

/home/yxz note Notice

$

其他情況,讀者可自行試之。

1. 註釋符

在shell程式設計中經常要對某些正文行進行註釋,以增加程式的可讀性。在Shell中以字元“#”開頭的正文行表示註釋行。

此外還有一些特殊字元如:用於輸入/輸出重定向與管道的、<>和|;執行後臺命令的&;命令執行運算子&&和||及表示命令組的{}將在下面各小節中加以介紹。

標準輸入/輸出和重定向

1. 標準輸入與輸出

我們知道,執行一個shell命令列時通常會自動開啟三個標準檔案,即標準輸入檔案(stdin),通常對應終端的鍵盤;標準輸出檔案(stdout)和標準錯誤輸出檔案(stderr),這兩個檔案都對應終端的螢幕。程式將從標準輸入檔案中得到輸入資料,將正常輸出資料輸出到標準輸出檔案,而將錯誤資訊送到標準錯誤檔案中。

我們以cat命令為例,cat命令的功能是從命令列給出的檔案中讀取資料,並將這些資料直接送到標準輸出。若使用如下命令:

$ cat config

將會把檔案config的內容依次顯示到螢幕上。但是,如果cat的命令列中沒有引數,它就會從標準輸入中讀取資料,並將其送到標準輸出。例如:

$ cat

Hello world

Hello world

Bye

Bye

$

使用者輸入的每一行都立刻被cat命令輸出到螢幕上。

另一個例子,命令sort按行讀入檔案正文(當命令列中沒有給出檔名時,表示從標準輸入讀入),將其排序,並將結果送到標準輸出。下面的例子是從標準輸入讀入一個採購單,並將其排序。

$ sort

bananas

carrots

apples

apples

bananas

carrots

$

這時我們在螢幕上得到了已排序的採購單。

直接使用標準輸入/輸出檔案存在以下問題:

輸入資料從終端輸入時,使用者費了半天勁輸入的資料只能用一次。下次再想用這些資料時就得重新輸入。而且在終端上輸入時,若輸入有誤修改起來不是很方便。

輸出到終端螢幕上的資訊只能看不能動。我們無法對此輸出作更多處理,如將輸出作為另一命令的輸入進行進一步的處理等。

為了解決上述問題,Linux系統為輸入、輸出的傳送引入了另外兩種機制,即輸入/輸出重定向和管道。

2. 輸入重定向

輸入重定向是指把命令(或可執行程式)的標準輸入重定向到指定的檔案中。也就是說,輸入可以不來自鍵盤,而來自一個指定的檔案。所以說,輸入重定向主要用於改變一個命令的輸入源,特別是改變那些需要大量輸入的輸入源。

例如,命令wc統計指定檔案包含的行數、單詞數和字元數。如果僅在命令列上鍵入:

$ wc

wc將等待使用者告訴它統計什麼,這時shell就好象死了一樣,從鍵盤鍵入的所有文字都出現在螢幕上,但並沒有什麼結果,直至按下<ctrl+d>,wc才將命令結果寫在螢幕上。

如果給出一個檔名作為wc命令的引數,如下例所示,wc將返回該檔案所包含的行數、單詞數和字元數。

$ wc /etc/passwd

20 23 726 /etc/passwd

$

另一種把/etc/passwd檔案內容傳給wc命令的方法是重定向wc的輸入。輸入重定向的一般形式為:命令

$ wc < /etc/passwd

20 23 726

$

另一種輸入重定向稱為here文件,它告訴shell當前命令的標準輸入來自命令列。here文件的重定向運算子使用<

$ wc<

>this text forms the content

>of the here document,which

>continues until the end of

>text delimter

>delim

4 17 98

在<

由於大多數命令都以引數的形式在命令列上指定輸入檔案的檔名,所以輸入重定向並不經常使用。儘管如此,當要使用一個不接受檔名作為輸入引數的命令,而需要的輸入內容又存在一個檔案裡時,就能用輸入重定向解決問題。

1. 輸出重定向

輸出重定向是指把命令(或可執行程式)的標準輸出或標準錯誤輸出重新定向到指定檔案中。這樣,該命令的輸出就不顯示在螢幕上,而是寫入到指定檔案中。

輸出重定向比輸入重定向更常用,很多情況下都可以使用這種功能。例如,如果某個命令的輸出很多,在螢幕上不能完全顯示,那麼將輸出重定向到一個檔案中,然後再用文字編輯器開啟這個檔案,就可以檢視輸出資訊;如果想儲存一個命令的輸出,也可以使用這種方法。還有,輸出重定向可以用於把一個命令的輸出當作另一個命令的輸入(還有一種更簡單的方法,就是使用管道,將在下面介紹)。

輸出重定向的一般形式為:命令>檔名。例如:

$ ls > directory.out

$ cat directory.out

ch1.doc ch2.doc ch3.doc chimp config mail/ test/

$

將ls命令的輸出儲存為一個名為directory.out的檔案。

注:如果>符號後邊的檔案已存在,那麼這個檔案將被重寫。

為避免輸出重定向中指定檔案只能存放當前命令的輸出重定向的內容,shell提供了輸出重定向的一種追加手段。輸出追加重定向與輸出重定向的功能非常相似,區別僅在於輸出追加重定向的功能是把命令(或可執行程式)的輸出結果追加到指定檔案的最後,而該檔案原有內容不被破壞。

如果要將一條命令的輸出結果追加到指定檔案的後面,可以使用追加重定向運算子>>。形式為:命令>>檔名。例如:

$ ls *.doc>>directory.out

$ cat directory.out

ch1.doc ch2.doc ch3.doc chimp config mail/ test/

ch1.doc ch2.doc ch3.doc

$

和程式的標準輸出重定向一樣,程式的錯誤輸出也可以重新定向。使用符號2>(或追加符號2>>)表示對錯誤輸出裝置重定向。例如下面的命令:

$ ls /usr/tmp 2> err.file

可在螢幕上看到程式的正常輸出結果,但又將程式的任何錯誤資訊送到檔案err.file中,以備將來檢查用。

還可以使用另一個輸出重定向運算子(&>)將標準輸出和錯誤輸出同時送到同一檔案中。例如:

$ ls /usr/tmp &> output.file

利用重定向將命令組合在一起,可實現系統單個命令不能提供的新功能。例如使用下面的命令序列:

$ ls /usr/bin > /tmp/dir

$ wc –w < /tmp/dir

459

統計了/usr/bin目錄下的檔案個數。

管 道

將一個程式或命令的輸出作為另一個程式或命令的輸入,有兩種方法,一種是透過一個臨時檔案將兩個命令或程式結合在一起,例如上個例子中的/tmp/dir檔案將ls和wc命令聯在一起;另一種是Linux所提供的管道功能。這種方法比前一種方法更好。

管道可以把一系列命令連線起來,這意味著第一個命令的輸出會作為第二個命令的輸入透過管道傳給第二個命令,第二個命令的輸出又會作為第三個命令的輸入,以此類推。顯示在螢幕上的是管道行中最後一個命令的輸出(如果命令列中未使用輸出重定向)。

透過使用管道符“|”來建立一個管道行。用管道重寫上面的例子:

$ ls /usr/bin|wc -w

1789

再如:

$ cat sample.txt|grep "High"|wc -l

管道將cat命令(列出一個檔案的內容)的輸出送給grep命令。grep命令在輸入裡查詢單詞High,grep命令的輸出則是所有包含單詞High的行,這個輸出又被送給wc命令,wc命令統計出輸入中的行數。假設sample.txt檔案的內容如下:

Things to do today:

Low:Go grocery shopping

High:Return movie

High:Clear level 3 in Alien vs. Predator

Medium:Pick up clothes from dry cleaner

那麼該管道行的結果是2。

命令替換

命令替換和重定向有些相似,但區別在於命令替換是將一個命令的輸出作為另外一個命令的引數。常用命令格式為:

command1 `command2`

其中,command2的輸出將作為command1的引數。需要注意的是這裡的`符號,被它括起來的內容將作為命令執行,執行後的結果作為command1的引數。例如:

$ cd `pwd`

該命令將pwd命令列出的目錄作為cd命令的引數,結果仍然是停留在當前目錄下。

第二十二課 在Bash中的操作      2000年/5月/29日

命令和檔名擴充套件特性

Bash命令列具有命令和檔名擴充套件特性。當輸入一個還沒完成的命令或檔名時,只需鍵入Tab鍵就能啟用命令和檔名擴充套件特性,從而完成該命令的剩餘輸入。如果有多個命令或檔案的字首相同,Bash將響鈴並等待使用者輸入足夠的字元,以便選擇唯一的命令或檔名,如果找到,系統將自動補齊搜尋到的命令或檔名,使用者按Enter鍵後,系統將執行這條指令。例如:

$ cat pre

$ cat preface

Bash也能列出當前目錄下部分匹配的檔名來完成檔名擴充套件。如果鍵入Esc,然後鍵入?,shell將列出所有與輸入的字串相匹配的檔名。例如下例,在沒有完成的輸入後鍵入Esc ?,shell將列出所有與輸入的字串相匹配的字串,然後shell回顯命令列,根據列出的檔名,可以鍵入要輸入的檔名或按下Tab鍵來完成檔名擴充套件。例如:

$ ls

document docudrama

$ cat doc

document

docudrama

$ cat docudrama

[例】下面是一個目錄包含的檔案列表:

Firebird2.7.tgz Firebird.README Firebird2.60.tgz

FireBird Firebird2.60.tgz.README

現在要刪除Firebird2.60.tgz.README檔案,鍵入:

$ rm –f Fi

系統會發出警報聲,並且自動將命令列補全為:

$ rm –f Fire

並等待使用者進一步輸入檔名的後面部分。現在再鍵入:

b

系統再次發出警報聲,並且自動將命令列補全為:

$ rm –f Firebird

並等待使用者進一步輸入檔名的後面部分。現在再鍵入:

2.6

系統再次發出警報聲,並且自動將命令列補全為:

$ rm –f Firebird2.60.tgz

並等待使用者進一步輸入檔名的後面部分。現在再鍵入:

.

此時命令將被補全為:

$ rm –f Firebird2.60.tgz..README

從上例可以看到,bash總是盡力根據使用者輸入的資訊來補全命令。當無法根據現有資訊補全命令時,則提示使用者再給出更多的資訊,然後再根據使用者的提示來進一步補全命令。作為使用者最好是能夠一次性給出足夠的資訊以便於bash進行命令補全;否則多按幾次,時間也就消耗掉了。

命令列編輯

在Bash中可以對命令列進行編輯,以便使用者在執行所鍵入的命令之前能夠修改所鍵入的命令。如果在鍵入命令時出現拼寫錯誤,只需在執行所鍵入的命令之前,使用編輯命令來糾正編輯錯誤,然後執行它,而不用重新輸入整行命令。這個功能對以長路徑檔名作引數的命令特別有用。

表10-2是對命令列編輯操作的一個總結。

表10-2 命令列編輯操作

 

命令列編輯操作

功能

Ctrl+b或左箭頭鍵

左移一個字元(移至前一個字元)

Ctrl+f或右箭頭鍵

右移一個字元(移至後一個字元)

Ctrl+a

移至行首

Ctrl+e

移至行尾

Esc b

左移一個單詞

Esc f

右移一個單詞

Del

刪除游標所在處的字元

Ctrl+d

刪除游標所在處的字元

BACKSPACE或Ctrl+h

刪除游標左邊的字元

Ctrl+k

刪除至行尾

 

命令歷史

在Bash中,history命令能夠儲存最近所執行的命令。這些命令的歷史記錄號從1開始,只有有限個命令可以被儲存起來,最多500個,即 history命令的歷史記錄號預設值為500。要檢視最近執行的命令,只要鍵入history命令,然後鍵入Enter鍵,最近執行過的命令即按先後順序被顯示出來(各條命令前的數字為歷史記錄號)。

[例】

$ history

1 cp mydata today

2 vi mydata

3 mv mydata reports

4 cd reports

5 ls

所有這些命令都被稱為事件(event),一個事件表示一個操作已經發生,即一個命令已被執行。這些事件根據它們被執行的先後順序用數字標識,這一標識稱為歷史事件號。最後執行的歷史事件的事件號最大。每個事件都可由它的歷史事件號或命令的初始字元或字串等確定。

利用history命令能夠查詢以前的事件,並可把它們顯示到命令列上執行這一事件。最簡便的方法就是利用上下箭頭鍵,把先前的事件逐次顯示到命令列。這個操作不需要執行history命令就可以執行。按動一下上箭頭鍵,那麼上一次執行的一個事件就將出現在命令列上,再按一下,上一次的前一事件又會出現在命令列上;按動一下下箭頭鍵,將會使當前事件的下一事件出現在命令列上。

Bash也可以透過鍵入Esc、Tab鍵來完成對歷史事件的字元擴充套件。和標準命令列擴充套件特性一樣,鍵入歷史事件的部分字串,然後鍵入Esc,再鍵入 Tab鍵,與剛才鍵入的字串相匹配的歷史事件將自動擴充套件並回顯到命令列處。如果不止一個事件與輸入的字串相匹配,就會聽到一聲響鈴,繼續鍵入字元或字串,shell將會唯一確定使用者所要鍵入的歷史事件。

還有一個查詢和執行歷史事件的命令——!命令。在!命令後鍵入與歷史事件相關聯的字元,這個關聯字元可以是歷史事件的歷史事件號,也可以是該事件的前幾個字元。在下面的例子中,查詢到歷史事件號為3的事件,然後又用其開頭的幾個字元去匹配,也查詢到該命令。

[例】

$ !3

mv mydata reports

$ !mv

mv mydata reports

也可以用一個偏移量(相對於歷史事件列表中最後一個事件)來查詢歷史事件。負的偏移量將從歷史事件列表表尾向前偏移。在下面的例子中,歷史事件號為2的事件“vi mydata”就是用一個負的偏移量查詢到的。必須注意的是,這個偏移量是相對於歷史事件列表中的最後一個事件的。在本例中,歷史事件列表中最後一個事件是事件5,歷史事件列表中第一個事件為1。從歷史事件號為5的事件,往前偏移4,即是歷史事件號為2的事件。

[例】

$ !-4

vi mydata

如果鍵入!!,則系統預設為上一事件。下面的例子中,使用者在命令列上鍵入!!命令,系統將執行上一事件:“ls”命令。

[例】

$ !!

ls

mydata today reports

也可以用“模式”來搜尋一個歷史事件。搜尋的“模式”必須用符號“?”括起來。下例是用“模式”“?myd?”來搜尋歷史事件號為3的歷史事件“vi mydata”。

[例】

$ !?myd?

vi mydata

1. 查詢歷史事件

可以在命令列上編輯歷史事件列表中的事件。表10-3列出了查詢歷史事件列表的各種操作。

表10-3 查詢歷史事件操作

查詢歷史事件操作

功能

Ctrl+n或向下游標鍵

移至歷史事件列表中當前事件的下一歷史事件

Ctrl+p或向上游標鍵

移至歷史事件列表中當前事件的前一歷史事件

Esc <

移至歷史事件列表表首

Esc >

移至歷史事件列表表尾

!event_num

用歷史事件號來定位一個歷史事件

!characters

用歷史事件的字元字首來查詢一個歷史事件

!?pattern

用“模式”來查詢歷史事件列表中的事件

!-event_num

透過偏移量來定位歷史事件

2. 配置history:HISTFILE及HISTSIZE

系統儲存的歷史事件數被儲存在一個特定的系統變數中,這個變數就是HISTSIZE。這個變數的預設值通常被設定為500。這個值可以被修改。例如:

$ HISTSIZE=10

將HISTSIZE的值重新設定為10。

歷史事件被儲存在一個檔案中,檔名由變數HISTFILE指定。通常這個檔案的預設名是.bash_history。透過給變數HISTFILE賦值,可以指定新的檔名。

[例】

$ echo $HISTFILE

/home/lisa/.bash_history

$ HISTFILE=”/home/lisa/newhist”

$ echo $HISTFILE

/home/lisa/newhist

以上操作先顯示變數HISTFILE的值,然後賦予它新的值“/home/lisa/newhist”,以後所有的歷史事件將被儲存在newhist檔案中。

別名

還有一個使工作變得輕鬆的方法是使用命令別名。命令別名通常是其他命令的縮寫,用來減少鍵盤輸入。

命令格式為:

alias [alias-name=’original-command’]

其中,alias-name是使用者給命令取的別名,original-command是原來的命令和引數。需要注意的是,由於Bash是以空格或者回車來識別原來的命令的,所以如果不使用引號就可能導致Bash只擷取第一個字,從而出現錯誤。如果alias命令後面不使用任何引數,則顯示當前正在使用的被別名化的命令及其別名。為命令取的別名在該次登入期間始終有效。如果使用者需要別名在每次登入時都有效,那麼就將alias命令寫到初始化指令碼檔案中。

[例]如果經常要鍵入如下的命令,最好為它建立一個別名來減少工作量。

$ cd /usr/X11/lib/X11

假如為這個長命令建立一個名為goconfig的別名,在Bash提示符下鍵入如下命令:

$ alias goconfig=’cd /usr/X11/lib/X11’

現在,除非您退出Bash,鍵入goconfig將和原來的長命令有同樣的作用。如果想取消別名,可以使用下面的命令:

$ unalias goconfig

這是一些很多人認為有用的別名,可以把它們寫入初始化指令碼檔案中來提高工作效率:

alias ll=’ls –l’

alias log=’logout’

alias ls=’ls –F’

如果您是一名DOS使用者並且習慣了DOS命令,可以用下面的別名定義使Linux表現得象DOS一樣:

alias dir=’ls’

alias copy=’cp’

alias rename=’mv’

alias md=’mkdir’

alias rd=’rmdir’

注意:在定義別名時,等號兩邊不能有空格,否則shell不能決定您需要做什麼。僅在命令中包含空格或特殊字元時才需要引號。

如果鍵入不帶任何引數的alias命令,將顯示所有已定義的別名。

提示符

Bash有兩級提示符。第一級提示符是經常見到的Bash在等待命令輸入時的情況。第一級提示符的預設值是$符號。如果使用者不喜歡這個符號,或者願意自己定義提示符,只需修改PS1變數的值。例如將其改為:

PS1=”Enter a command:”

第二級提示符是當Bash為執行某條命令需要使用者輸入更多資訊時顯示的。第二級提示符預設為>。如果需要自己定義該提示符,只需改變PS2變數的值。例如將其改為:

PS2=”More information:”

上面的兩個例子都是設定提示符為靜態字串的情況。其實使用者也可以使用一些事先已經定義好的特殊字元。這些特殊字元將使提示符中包含當前時間之類的資訊。表10-4列出了最常用的一些特殊字元及其含義。

表10-4 bash提示符常用特殊字元

 

特殊字元

說 明

!

顯示該命令的歷史編號

#

顯示shell啟用後,當前命令的歷史編號

$

顯示一個$符號,如果當前使用者是root則顯示#符號

顯示一個反斜槓

d

顯示當前日期

h

顯示執行該shell的計算機主機名

n

列印一個換行符,這將導致提示符跨行

s

顯示正在執行的Shell的名稱

t

顯示當前時間

u

顯示當前使用者的使用者名稱

W

顯示當前工作目錄基準名

w

顯示當前工作目錄

 

這些特殊字元可以組合起來,為使用者提供一些提示符,提供很有用的資訊。下面來看幾個實際例子:

PS1=”t”

將使提示符變成如下所示:

02:16:15

而 PS1=t

將使提示符變成如下所示:

t

若PS1=”t”

將使提示符變成如下所示:

02:16:30

該例就是使用兩個特殊字元的組合得到的。

控制shell的執行方式

Bash有一些特殊變數,能控制shell以不同的方式工作。例如,變數noclobber能防止在重定向輸出時意外地覆蓋一個檔案。透過set命令可以設定noclobber變數的有效或無效。set命令有兩個引數:一個是指定變數開(on)或關(off)的選項,一個是特殊變數的變數名。要使某一特殊變數開(有效),用-o選項,要使其關(無效),用+o選項。例如:

$ set –o noclobber // 使noclobber變數開

$ set +o noclobber // 使noclobber變數關

三個最常用的shell特殊變數有:ignoreeof、noclobber及noglob。

ignoreeof

ignoreeof變數用來禁止使用ctrl+d來退出shell(ctrl+d不僅用來退出shell,而且可以終止使用者直接輸往標準輸出上的輸入。該操作經常在一些shell實用命令中使用,例如實用命令cat。在這些實用程式操作中,非常容易誤操作而意外地退出shell。ignoreeof特殊變數正是用來防止這種意外的退出。例如:

$ set –o ignoreeof

之後,使用者只能用logout或exit命令退出shell。

noclobber

noclobber變數可以在重定向輸出時保護已存在的檔案,防止被意外地覆蓋。在下例中,使用者設定noclobber為有效,在重定向時,使用者試圖去覆蓋已經存在的檔案myfile,此時系統將返回一個錯誤資訊。

[例]

$ set –o noclobber

$ cat preface>myfile

bash: myfile: cannot overwrite existing file

$

noglob

設定noglob變數後,shell將不擴充套件檔名中一些特殊的字元或字串。如字元*、?、[]等將不再作為萬用字元。如果使用者希望列出結尾為?的檔名answer?,可透過如下步驟:首先,使用者使noglob變數為無效,然後再列出檔名。可以看到,目前命令列上的問號?被認為是檔名中的一個字元,而不再被看作萬用字元。

$ set –o noglob

$ ls answer?

answer?

子shell與export命令

使用者登入到Linux系統後,系統將啟動一個使用者shell。在這個shell中,可以使用shell命令或宣告變數,也可以建立並執行shell指令碼程式。執行shell指令碼程式時,系統將建立一個子shell。此時,系統中將有兩個shell,一個是登入時系統啟動的shell,另一個是系統為執行指令碼程式建立的shell。當一個指令碼程式執行完畢,它的指令碼shell將終止,可以返回到執行該指令碼之前的shell。從這種意義上來說,使用者可以有許多 shell,每個shell都是由某個shell(稱為父shell)派生的。

在子shell中定義的變數只在該子shell內有效。如果在一個shell指令碼程式中定義了一個變數,當該指令碼程式執行時,這個定義的變數只是該指令碼程式內的一個區域性變數,其他的shell不能引用它,要使某個變數的值可以在其他shell中被改變,可以使用export命令對已定義的變數進行輸出。 export命令將使系統在建立每一個新的shell時定義這個變數的一個複製。這個過程稱之為變數輸出。

[例]在本例中,變數myfile是在dispfile指令碼程式中定義的。然後用export命令將變數myfile輸出至任何子shell,例如當執行printfile指令碼程式時產生的子shell。

dispfile指令碼程式清單:

/**************begin dispfile**************/

myfile=”List”

export myfile

echo “Displaying $myfile”

pr –t –n $myfile

printfile

/**************end dispfile***************/

 

printfile指令碼程式清單:

/**************begin printfile**************/

echo “Printing $myfile”

lpr $myfile&

/**************end printfile**************/

$dispfile

Displaying List

1 screen

2 modem

3 paper

Printing List

$

定製Bash

在本節中已經介紹了很多定製Bash的方法,但是迄今為止,這些方法都只是對當前Bash對話有用。只要使用者退出登入,所做的一切改變都會丟失。所以應該在Bash的初始化檔案中做永久性的修改。

使用者可以將每次啟動Bash所需要執行的命令放入初始化檔案中,最常見的命令就是alias命令和變數定義兩種。系統中的每個使用者在其主目錄中都有一個.bash_profile檔案,Bash每次啟動時都將讀取該檔案,其中包含的所有命令都將被執行。

下面便是預設.bash_profile檔案的程式碼:

#.bash_profile

#Get the aliases and functions

if [-f ~/.bashrc ];then

.~/.bashrc

fi

#User specific environment and startup programs

PATH=$PATH:$HOME/bin

ENV=$HOME/.bashrc

USERNAME=””

Export USERNAME ENV PAT

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/8225414/viewspace-944794/,如需轉載,請註明出處,否則將追究法律責任。

相關文章