BASH的基本語法
·最簡單的例子—— Hello World!
·關于輸入、輸出和錯誤輸出
·BASH中對變量的規定(與C語言的異同)
·BASH中的基本流程控制語法
·函數的使用
2.1最簡單的例子—— Hello World!
幾乎所有的講解編程的書給讀者的第一個例子都是Hello World程序,那么我們今天也就從這個例子出發,來逐步了解BASH。
用vi編輯器編輯一個hello文件如下:
#!/bin/bash
# This is a very simple example
echo Hello World
這樣最簡單的一個BASH程序就編寫完了。這里有幾個問題需要說明一下:
一,第一行的#!是什么意思
二,第一行的/bin/bash又是什么意思
三,第二行是注釋嗎
四,echo語句
五,如何執行該程序
#!是說明hello這個文件的類型的,有點類似于Windows系統下用不同文件后綴來表示不同文件類型的意思(但不相同)。Linux系統根據"#!"及該字串后面的信息確定該文件的類型,關于這一問題同學們回去以后可以通過"man magic"命令 及/usr/share/magic文件來了解這方面的更多內容。在BASH中 第一行的"#!"及后面的"/bin/bash"就表明該文件是一個BASH程序,需要由/bin目錄下的bash程序來解釋執行。BASH這個程序一般是存放在/bin目錄下,如果你的Linux系統比較特別,bash也有可能被存放在/sbin、/usr/local/bin、/usr/bin、/usr/sbin或/usr/local/sbin這樣的目錄下;如果還找不到,你可以用"locate bash" "find / -name bash 2> /dev/null"或"whereis bash"這三個命令找出bash所在的位置;如果仍然找不到,那你可能需要自己動手安裝一個BASH軟件包了。
第二行的"# This is a ..."就是BASH程序的注釋,在BASH程序中從“#”號(注意:后面緊接著是“!”號的除外)開始到行尾的多有部分均被看作是程序的注釋。的三行的echo語句的功能是把echo后面的字符串輸出到標準輸出中去。由于echo后跟的是"Hello World"這個字符串,因此"Hello World"這個字串就被顯示在控制臺終端的屏幕上了。需要注意的是BASH中的絕大多數語句結尾處都沒有分號。
如何執行該程序呢?有兩種方法:一種是顯式制定BASH去執行:
$ bash hello或
$ sh hello(這里sh是指向bash的一個鏈接,“lrwxrwxrwx 1 root root 4 Aug 20 05:41 /bin/sh -> bash”)
或者可以先將hello文件改為可以執行的文件,然后直接運行它,此時由于hello文件第一行的"#! /bin/bash"的作用,系統會自動用/bin/bash程序去解釋執行hello文件的:
$ chmod u+x hello
$ ./hello
此處沒有直接“$ hello”是因為當前目錄不是當前用戶可執行文件的默認目錄,而將當前目錄“.”設為默認目錄是一個不安全的設置。
需要注意的是,BASH程序被執行后,實際上Linux系統是另外開設了一個進程來運行的。
2.2關于輸入、輸出和錯誤輸出
在字符終端環境中,標準輸入/標準輸出的概念很好理解。輸入即指對一個應用程序 或命令的輸入,無論是從鍵盤輸入還是從別的文件輸入;輸出即指應用程序或命令產生的一些信息;與Windows系統下不同的是,Linux系統下還有一個標準錯誤輸出的概念,這個概念主要是為程序調試和系統維護目的而設置的,錯誤輸出于標準輸出分開可以讓一些高級的錯誤信息不干擾正常的輸出信息,從而方便一般用戶的使用。
在Linux系統中:標準輸入(stdin)默認為鍵盤輸入;標準輸出(stdout)默認為屏幕輸出;標準錯誤輸出(stderr)默認也是輸出到屏幕(上面的std表示standard)。在BASH中使用這些概念時一般將標準輸出表示為1,將標準錯誤輸出表示為2。下面我們舉例來說明如何使用他們,特別是標準輸出和標準錯誤輸出。
輸入、輸出及標準錯誤輸出主要用于I/O的重定向,就是說需要改變他們的默認設置。先看這個例子:
$ ls > ls_result$ ls -l >> ls_result
上面這兩個命令分別將ls命令的結果輸出重定向到ls_result文件中和追加到ls_result文件中,而不是輸出到屏幕上。">"就是輸出(標準輸出和標準錯誤輸出)重定向的代表符號,連續兩個">"符號,即">>"則表示不清除原來的而追加輸出。下面再來看一個稍微復雜的例子:
$ find /home -name lost* 2> err_result
這個命令在">"符號之前多了一個"2","2>"表示將標準錯誤輸出重定向。由于/home目錄下有些目錄由于權限限制不能訪問,因此會產生一些標準錯誤輸出被存放在err_result文件中。大家可以設想一下find /home -name lost* 2>>err_result命令會產生什么結果?
如果直接執行find /home -name lost* > all_result,其結果是只有標準輸出被存入all_result文件中,要想讓標準錯誤輸出和標準輸入一樣都被存入到文件中,那該怎么辦呢?看下面這個例子:
$ find /home -name lost* > all_result 2>& 1
上面這個例子中將首先將標準錯誤輸出也重定向到標準輸出中,再將標準輸出重定向到all_result這個文件中。這樣我們就可以將所有的輸出都存儲到文件中了。為實現上述功能,還有一種簡便的寫法如下:
$ find /home -name lost* >& all_result
如果那些出錯信息并不重要,下面這個命令可以讓你避開眾多無用出錯信息的干擾:
$find /home -name lost* 2> /dev/null
同學們回去后還可以再試驗一下如下幾種重定向方式,看看會出什么結果,為什么?
$ find /home -name lost* > all_result 1>& 2
$ find /home -name lost* 2> all_result 1>& 2
$ find /home -name lost* 2>& 1 > all_result
另外一個非常有用的重定向操作符是"-",請看下面這個例子:
$ (cd /source/directory && tar cf - . ) | (cd /dest/directory && tar xvfp -)
該命令表示把/source/directory目錄下的所有文件通過壓縮和解壓,快速的全部移動到/dest/directory目錄下去,這個命令在/source/directory和/dest/directory不處在同一個文件系統下時將顯示出特別的優勢。
下面還幾種不常見的用法:
n<&-表示將n號輸入關閉
<&-表示關閉標準輸入(鍵盤)
n>&-表示將n號輸出關閉
>&-表示將標準輸出關閉
2.3???? BASH中對變量的規定(與C語言的異同)
好了下面我們進入正題,先看看BASH中的變量是如何定義和使用的。對于熟悉C語言的程序員,我們將解釋BASH中的定義和用法與C語言中有何不同。
2.3.1. BASH中的變量介紹
我們先來從整體上把握一下BASH中變量的用法,然后再去分析BASH中變量使用與C語言中的不同。BASH中的變量都是不能含有保留字,不能含有"-"等保留字符,也不能含有空格。
2.3.1.1簡單變量
在BASH中變量定義是不需要的,沒有"int i"這樣的定義過程。如果想用一個變量,只要他沒有在前面被定義過,就直接可以用,當然你使用該變量的第一條語句應該是對他賦初值了,如果你不賦初值也沒關系,只不過該變量是空( 注意:是NULL,不是0)。不給變量賦初值雖然語法上不反對,但不是一個好的編程習慣。好了我們看看下面的例子:
首先用vi編輯下面這個文件 hello2:
#!/bin/bash
# give the initialize value to STR
STR="Hello World"
echo $STR
在上面這個程序中我們需要注意下面幾點:
一,變量賦值時,'='左右兩邊都不能有空格;
二,BASH中的語句結尾不需要分號(";");
三,除了在變量賦值和在FOR循環語句頭中,BASH中的變量使用必須在變量前加"$"符號,同學們可以將上面程序中第三行改為"echo STR"再試試,看看會出什么結果。==>output: STR
四,由于BASH程序是在一個新的進程中運行的,所以該程序中的變量定義和賦值不會改變其他進程或原始Shell中同名變量的值,也不會影響他們的運行。
更細致的文檔甚至提到以但引號括起來的變量將不被BASH解釋為變量,如'$STR',而被看成為純粹的字符串。而且更為標準的變量引用方式是${STR}這樣的,$STR自不過是對${STR}的一種簡化。在復雜情況下(即有可能產生歧義的地方)最好用帶{}的表示方式。
BASH中的變量既然不需要定義,也就沒有類型一說,一個變量即可以被定義為一個字符串,也可以被再定義為整數。如果對該變量進行整數運算,他就被解釋為整數;如果對他進行字符串操作,他就被看作為一個字符串。請看下面的例子:
#!/bin/bash
x=1999
let "x = $x + 1"
echo $x
x="olympic'"$x
echo $x
關于整數變量計算,有如下幾種:" + - * / % ",他們的意思和字面意思相同。整數運算一般通過let和expr這兩個指令來實現,如對變量x加1可以寫作:let"x = $x + 1"或者x=`expr $x + 1`
在比較操作上,整數變量和字符串變量各不相同,詳見下表:
比如:
比較字符串a和b是否相等就寫作:if [ $a = $b ]
判斷字符串a是否為空就寫作:if [ -z $a ]
判斷整數變量a是否大于b就寫作:if [ $a -gt $b ]
更細致的文檔推薦在字符串比較時盡量不要使用-n,而用! -z來代替。(其中符號"!"表示求反操作)
BASH中的變量除了用于對 整數 和 字符串 進行操作以外,另一個作用是作為文件變量。BASH是Linux操作系統的Shell,因此系統的文件必然是BASH需要操作的重要對象,如if [ -x /root ]可以用于判斷/root目錄是否可以被當前用戶進入。下表列出了BASH中用于判斷文件屬性的操作符:
注意:上表中的file及file1、file2都是指某個文件或目錄的路徑。
2.3.1.1.關于局部變量
在BASH程序中如果一個變量被使用了,那么直到該程序的結尾,該變量都一直有效。為了使得某個變量存在于一個局部程序塊中,就引入了局部變量的概念。BASH中,在變量首次被賦初值時加上local關鍵字就可以聲明一個局部變量,如下面這個例子:
#!/bin/bash
HELLO=Hello
function hello {
local HELLO=World
echo $HELLO
}
echo $HELLO
hello
echo $HELLO
該程序的執行結果是:
Hello
World
Hello
這個執行結果表明全局變量$HELLO的值在執行函數hello時并沒有被改變。也就是說局部變量$HELLO的影響只存在于函數那個程序塊中。
2.3.2. BASH中的變量與C語言中變量的區別
這里我們為原來不熟悉BASH編程,但是非常熟悉C語言的程序員總結一下在BASH環境中使用變量需要注意的問題。
1,BASH中的變量在引用時都需要在變量前加上"$"符號(第一次賦值及在For循環的頭部不用加"$"符號);
2,BASH中沒有浮點運算,因此也就沒有浮點類型的變量可用;
3,BASH中的整形變量的比較符號與C語言中完全不同,而且整形變量的算術運算也需要經過let或expr語句來處理;
2.4???? BASH中的基本流程控制語法
BASH中幾乎含有C語言中常用的所有控制結構,如條件分支、循環等,下面逐一介紹。
2.4.1if...then...else
if語句用于判斷和分支,其語法規則和C語言的if非常相似。其幾種基本結構為:
if [ expression ]
then
statments
fi
或者
if [ expression ]
then
statments
else
statments
fi
或者
if [ expression ]
then
statments
else if [ expression ]
then
statments
else
statments
fi
或者
if [ expression ]
then
statments
elif [ expression ]
then
statments
else
statments
fi
值得說明的是如果你將if和then簡潔的寫在一行里面,就必須在then前面加上分號,如:if [ expression ]; then ...。下面這個例子說明了如何使用if條件判斷語句:
#!/bin/bash
if [ $1 -gt 90 ]
then
echo "Good, $1"
elif [ $1 -gt 70 ]
then
echo "OK, $1"
else
echo "Bad, $1"
fi
exit 0
上面例子中的$1是指命令行的第一個參數,這個會在后面的“BASH中的特殊保留字”中講解。
2.4.2for
for循環結構與C語言中有所不同,在BASH中for循環的基本結構是:
for $var in
do
statments
done
其中$var是循環控制變量,
是$var需要遍歷的一個集合,do/done對包含了循環體,相當于C語言中的一對大括號。另外如果do和for被寫在同一行,必須在do前面加上";"。如:for $var in
; do。下面是一個運用for進行循環的例子:
#!/bin/bash
for day in Sun Mon Tue Wed Thu Fri Sat
do
echo $day
done
#如果列表被包含在一對雙引號中,則被認為是一個元素
for day in "Sun Mon Tue Wed Thu Fri Sat"
do
echo $day
done
exit 0
注意上面的例子中,在for所在那行的變量day是沒有加"$"符號的,而在循環體內,echo所在行變量$day是必須加上"$"符號的。另外如果寫成for day而沒有后面的in
部分,則day將取遍命令行的所有參數。如這個程序:
#!/bin/bash
for param
do
echo $param
done
exit 0
上面這個程序將列出所有命令行參數。for循環結構的循環體被包含在do/done對中,這也是后面的while、until循環所具有的特點。
2.4.3while
while循環的基本結構是:
while [ condition ]
do
statments
done
這個結構請大家自己編寫一個例子來驗證。
2.4.4until
until循環的基本結構是:
until [ condition is TRUE ]
do
statments
done
這個結構也請大家自己編寫一個例子來驗證。
2.4.5case
BASH中的case結構與C語言中的switch語句的功能比較類似,可以用于進行多項分支控制。其基本結構是:
case "$var" in
condition1 )
statments1;;
condition2 )
statments2;;
...
* )
default statments;;
esac
下面這個程序是運用case結構進行分支執行的例子:
#!/bin/bash
echo "Hit a key, then hit return."
read Keypress
case "$Keypress" in
[a-z] ) echo "Lowercase letter";;
[A-Z] ) echo "Uppercase letter";;
[0-9] ) echo "Digit";;
* ) echo "Punctuation, whitespace, or other";;
esac
exit 0
上面例子中的第四行"read Keypress"一句中的read語句表示從鍵盤上讀取輸入。這個命令將在本講義的BASH的其他高級問題中講解。
2.4.6break/continue
熟悉C語言編程的都很熟悉break語句和continue語句。BASH中同樣有這兩條語句,而且作用和用法也和C語言中相同,break語句可以讓程序流程從當前循環體中完全跳出,而continue語句可以跳過當次循環的剩余部分并直接進入下一次循環。
2.5函數的使用
BASH是一個相對簡單的腳本語言,不過為了方便結構化的設計,BASH中也提供了函數定義的功能。BASH中的函數定義很簡單,只要向下面這樣寫就可以了:
function my_funcname {
code block
}
或者
my_funcname() {
code block
}
上面的第二種寫法更接近于C語言中的寫法。BASH中要求函數的定義必須在函數使用之前,這是和C語言用頭文件說明函數方法的不同。
更進一步的問題是如何給函數傳遞參數和獲得返回值。BASH中函數參數的定義并不需要在函數定義處就制定,而只需要在函數被調用時用BASH的保留變量$1 $2 ...來引用就可以了;BASH的返回值可以用return語句來指定返回一個特定的整數,如果沒有return語句顯式的返回一個返回值,則返回值就是該函數最后一條語句執行的結果(一般為0,如果執行失敗返回錯誤碼)。函數的返回值在調用該函數的程序體中通過$?保留字來獲得。下面我們就來看一個用函數來計算整數平方的例子:
#!/bin/bash
square() {
let "res = $1 * $1"
return $res
}
square $1
result=$?
echo $result
exit 0
BASH中的特殊保留字
·保留變量
·隨機數
·運算符
·變量的特殊操作
3.1保留變量
BASH中有一些保留變量,下面列出了一些:
$IFS這個變量中保存了用于分割輸入參數的分割字符,默認識空格。
$HOME這個變量中存儲了當前用戶的根目錄路徑。
$PATH這個變量中存儲了當前Shell的默認路徑字符串。
$PS1表示第一個系統提示符。
$PS2表示的二個系統提示符。
$PWD表示當前工作路徑。
$EDITOR表示系統的默認編輯器名稱。
$BASH表示當前Shell的路徑字符串。
$0, $1, $2, ...
表示系統傳給腳本程序或腳本程序傳給函數的第0個、第一個、第二個等參數。
$#表示腳本程序的命令參數個數或函數的參數個數。
$$表示該腳本程序的進程號,常用于生成文件名唯一的臨時文件。
$?表示腳本程序或函數的返回狀態值,正常為0,否則為非零的錯誤號。
$*表示所有的腳本參數或函數參數。
$@和$*涵義相似,但是比$*更安全。
$!表示最近一個在后臺運行的進程的進程號。
3.2隨機數
隨機數是經常要用到的,BASH中也提供了這個功能,請看下面這個程序:
#!/bin/bash
# Prints different random integer from 1 to 65536
a=$RANDOM
echo $a
exit 0
這個程序可以在每次執行的時候隨機的打印出一個大小在1到65536之間的整數。
3.3運算符
算術運算符
+ - * / %表示加減乘除和取余運算
+= -= *= /=同C語言中的含義
位操作符
<< <<= >> >>=表示位左右移一位操作
& &= | |=表示按位與、位或操作
~ !表示非操作
^ ^=表示異或操作
關系運算符
< > <= >= == !=表示大于、小于、大于等于、小于等于、等于、不等于操作
&& ||邏輯與、邏輯或操作
3.4變量的特殊操作
BASH中還有一些對變量的簡潔、快速的操作,大家還記得"${var}"和"$var"同樣是對變量的引用吧,對${var}進行一些變化就可以產生一些新功能:
${var-default}表示如果變量$var還沒有設置,則保持$var沒有設置的狀態,并返回后面的默認值default。
${var=default}表示如果變量$var還沒有設置,則取后面的默認值default。
${var+otherwise}表示如果變量$var已經設置,則返回otherwise的值,否則返回空( null )。
${var?err_msg}表示如果變量$var已經設置,則返回該變量的值,否則將后面的err_msg輸出到標準錯誤輸出上。
請同學們自己嘗試下面的例子:
#!/bin/bash
echo ${var?There is an error}
exit 0
還有下面一些用法,這些用法主要用于從文件路徑字符串中提取有用信息:
${var#pattern}, ${var##pattern}用于從變量$var中剝去最短(最長)的和pattern相匹配的最左側的串。
${var%pattern}, ${var%%pattern}用于從變量$var中剝去最短(最長)的和pattern相匹配的最右側的串。
另外BASH 2中還加入下面一些操作:
${var:pos}表示去掉變量$var中前pos個字符。
${var:pos:len}表示變量$var中去掉前pos個字符后的剩余字符串的前len個字符。
${var/pattern/replacement}表示將變量$var中第一個出現的pattern模式替換為replacement字符串。
${var//pattern/replacement}表示將變量$var中出現的所有pattern模式全部都替換為replacment字符串。
BASH中的其他高級問題
·BASH中對返回值的處理
·用BASH設計簡單用戶界面
·在BASH中讀取用戶輸入
·一些特殊的慣用法
·BASH程序的調試
·關于BASH2
4.1???? BASH中對返回值的處理
無論是在Shell中對BASH腳本返回值的處理,還是在腳本中對函數返回值的處理,都是通過"$?"系統變量來獲得。BASH要求返回值必須為一個整數,不能用return語句返回字符串變量。
4.2用BASH設計簡單用戶界面
BASH中提供了一個小的語句格式,可以讓程序快速的設計出一個字符界面的用戶交互選擇的菜單,該功能就是由select語句來實現的,select語句的語法為:
select var in
do
statments use $var
done
上面的語法結構在執行后,BASH會將
中的所有項加上數字列在屏幕上等待用戶選擇,在用戶作出選擇后,變量$var中就包含了那個被選中的字符串,然后就可以對該變量進行需要的操作了。我們可以從下面的例子中更直觀的來理解這個功能:
#!/bin/bash
OPTIONS="Hello Quit"
select opt in $OPTIONS; do
if [ "$opt" = "Quit" ]; then
echo done
exit
elif [ "$opt" = "Hello" ]; then
echo Hello World
else
clear
echo bad option
fi
done
exit 0
大家可以試著執行上面的程序,看看是什么執行結果。
4.3在BASH中讀取用戶輸入
BASH中通過read函數來實現讀取用戶輸入的功能,如下面這段程序:
#!/bin/bash
echo Please enter your name
read NAME
echo "Hi! $NAME !"
exit 0
上面這個腳本讀取用戶的輸入,并回顯在屏幕上。
另外BASH中還提供另外一種稱為here documents的結構????,可以將用戶需要通過鍵盤輸入的字符串改為從程序體中直接讀入,如密碼。下面的小程序演示了這個功能:
#!/bin/bash
passwd="aka@tsinghua"
ftp -n localhost
binary
bye
FTPFTP
exit 0
這個程序在用戶需要通過鍵盤敲入一些字符時,通過程序內部的動作來模擬鍵盤輸入。請注意here documents的基本結構為:
command
...
SOMESPECIALSTRING
這里要求在需要鍵盤輸入的命令后,直接加上<<符號,然后跟上一個特別的字符串,在該串后按順序輸入本來應該由鍵盤輸入的所有字符,在所有需要輸入的字符都結束后,重復一遍前面<<符號后的“特別的字符串”即表示該輸入到此結束。
4.4一些特殊的慣用法
在BASH中()一對括號一般被用于求取括號中表達式的值或命令的執行結果,如:(a=hello; echo $a),其作用相當于`...`。
:有兩個含義,一是表示空語句,有點類似于C語言中的單個";"。表示該行是一個空命令,如果被用在while/until的頭結構中,則表示值0,會使循環一直進行下去,如下例:
while :
do
operation-1
operation-2
...
operation-n
done
另外:還可以用于求取后面變量的值,比如:
#!/bin/bash
: ${HOSTNAME?} {USER?} {MAIL?}
echo $HOSTNAME
echo $USER
echo $MAIL
exit 0
在BASH中export命令用于將系統變量輸出到外層的Shell中了。
4.5 BASH程序的調試
用bash -x bash-script命令,可以查看一個出錯的BASH腳本到底錯在什么地方,可以幫助程序員找出腳本中的錯誤。
另外用trap語句可以在BASH腳本出錯退出時打印出一些變量的值,以供程序員檢查。trap語句必須作為繼"#!/bin/bash"后的第一句非注釋代碼,一般trap命令被寫作:trap 'message $checkvar1 $checkvar2' EXIT。
4.6關于BASH2
使用bash -version命令可以看出當前你正在使用的BASH是什么版本,一般版本號為1.14或其他版本。而現在機器上一般還安裝了一個版本號為2.0的BASH。該BASH也在/bin目錄下。BASH2提供了一些新功能,有興趣的同敘可以自己去看相關資料,或直接man bash2即可。