最簡單的例子 —— Hello World!html
關於輸入、輸出和錯誤輸出git
BASH 中對變量的規定(與 C 語言的異同)程序員
BASH 中的基本流程控制語法express
函數的使用編程
幾乎全部的講解編程的書給讀者的第一個例子都是 Hello World 程序,那麼咱們今天也就從這個例子出發,來逐步瞭解 BASH。小程序
用 vi 編輯器編輯一個 hello 文件以下:
#!/bin/bash
# This is a very simple example
echo Hello World安全
這樣最簡單的一個 BASH 程序就編寫完了。這裏有幾個問題須要說明一下:bash
一,第一行的 #! 是什麼意思
二,第一行的 /bin/bash 又是什麼意思
三,第二行是註釋嗎
四,echo 語句
五,如何執行該程序dom
#! 是說明 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 系統是另外開設了一個進程來運行的。
在字符終端環境中,標準輸入/標準輸出的概念很好理解。輸入即指對一個應用程序 或命令的輸入,不管是從鍵盤輸入仍是從別的文件輸入;輸出即指應用程序或命令產生的一些信息;與 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 號輸出關閉
>&- 表示將標準輸出關閉
好了下面咱們進入正題,先看看 BASH 中的變量是如何定義和使用的。對於熟悉 C 語言的程序員,咱們將解釋 BASH 中的定義和用法與 C 語言中有何不一樣。
咱們先來從總體上把握一下 BASH 中變量的用法,而後再去分析 BASH 中變量使用與 C 語言中的不一樣。BASH 中的變量都是不能含有保留字,不能含有 "-" 等保留字符,也不能含有空格。
在 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`
在比較操做上,整數變量和字符串變量各不相同,詳見下表:
對應的操做
|
整數操做
|
字符串操做
|
相同
|
-eq
|
=
|
不一樣
|
-ne
|
!=
|
大於
|
-gt
|
>
|
小於
|
-lt
|
<
|
大於或等於
|
-ge
|
|
小於或等於
|
-le
|
|
爲空
|
-z
|
|
不爲空
|
-n
|
好比:
比較字符串 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 中用於判斷文件屬性的操做符:
運算符
|
含義( 知足下面要求時返回 TRUE ) |
-e file
|
文件 file 已經存在 |
-f file
|
文件 file 是普通文件 |
-s file
|
文件 file 大小不爲零 |
-d file
|
文件 file 是一個目錄 |
-r file
|
文件 file 對當前用戶能夠讀取 |
-w file
|
文件 file 對當前用戶能夠寫入 |
-x file
|
文件 file 對當前用戶能夠執行 |
-g file
|
文件 file 的 GID 標誌被設置 |
-u file
|
文件 file 的 UID 標誌被設置 |
-O file
|
文件 file 是屬於當前用戶的 |
-G file
|
文件 file 的組 ID 和當前用戶相同 |
file1 -nt file2
|
文件 file1 比 file2 更新 |
file1 -ot file2
|
文件 file1 比 file2 更老 |
注意:上表中的 file 及 file一、file2 都是指某個文件或目錄的路徑。
在 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 的影響只存在於函數那個程序塊中。
這裏咱們爲原來不熟悉 BASH 編程,可是很是熟悉 C 語言的程序員總結一下在 BASH 環境中使用變量須要注意的問題。
1,BASH 中的變量在引用時都須要在變量前加上 "$" 符號( 第一次賦值及在For循環的頭部不用加 "$"符號 );
2,BASH 中沒有浮點運算,所以也就沒有浮點類型的變量可用;
3,BASH 中的整形變量的比較符號與 C 語言中徹底不一樣,並且整形變量的算術運算也須要通過 let 或 expr 語句來處理;
BASH 中幾乎含有 C 語言中經常使用的全部控制結構,如條件分支、循環等,下面逐一介紹。
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 中的特殊保留字」中講解。
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 循環所具備的特色。
while 循環的基本結構是:
while [ condition ]
do
statments
done
這個結構請你們本身編寫一個例子來驗證。
until 循環的基本結構是:
until [ condition is TRUE ]
do
statments
done
這個結構也請你們本身編寫一個例子來驗證。
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 的其餘高級問題中講解。
熟悉 C 語言編程的都很熟悉 break 語句和 continue 語句。BASH 中一樣有這兩條語句,並且做用和用法也和 C 語言中相同,break 語句可讓程序流程從當前循環體中徹底跳出,而 continue 語句能夠跳過當次循環的剩餘部分並直接進入下一次循環。
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 中有一些保留變量,下面列出了一些:
$IFS 這個變量中保存了用於分割輸入參數的分割字符,默認識空格。
$HOME 這個變量中存儲了當前用戶的根目錄路徑。
$PATH 這個變量中存儲了當前 Shell 的默認路徑字符串。
$PS1 表示第一個系統提示符。
$PS2 表示的二個系統提示符。
$PWD 表示當前工做路徑。
$EDITOR 表示系統的默認編輯器名稱。
$BASH 表示當前 Shell 的路徑字符串。
$0, $1, $2, ...
表示系統傳給腳本程序或腳本程序傳給函數的第0個、第一個、第二個等參數。
$# 表示腳本程序的命令參數個數或函數的參數個數。
$$ 表示該腳本程序的進程號,經常使用於生成文件名惟一的臨時文件。
$? 表示腳本程序或函數的返回狀態值,正常爲 0,不然爲非零的錯誤號。
$* 表示全部的腳本參數或函數參數。
$@ 和 $* 涵義類似,可是比 $* 更安全。
$! 表示最近一個在後臺運行的進程的進程號。
隨機數是常常要用到的,BASH 中也提供了這個功能,請看下面這個程序:
#!/bin/bash
# Prints different random integer from 1 to 65536
a=$RANDOM
echo $a
exit 0
這個程序能夠在每次執行的時候隨機的打印出一個大小在 1 到 65536 之間的整數。
算術運算符
+ - * / % 表示加減乘除和取餘運算
+= -= *= /= 同 C 語言中的含義
位操做符
<< <<= >> >>= 表示位左右移一位操做
& &= | |= 表示按位與、位或操做
~ ! 表示非操做
^ ^= 表示異或操做
關係運算符
< > <= >= == != 表示大於、小於、大於等於、小於等於、等於、不等於操做
&& || 邏輯與、邏輯或操做
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 程序的調試
關於 BASH2
不管是在 Shell 中對 BASH 腳本返回值的處理,仍是在腳本中對函數返回值的處理,都是經過 "$?" 系統變量來得到。BASH 要求返回值必須爲一個整數,不能用 return 語句返回字符串變量。
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
你們能夠試着執行上面的程序,看看是什麼執行結果。
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 <<FTPFTP
user anonymous $passwd
binary
bye
FTPFTP
exit 0
這個程序在用戶須要經過鍵盤敲入一些字符時,經過程序內部的動做來模擬鍵盤輸入。請注意 here documents 的基本結構爲:
command <<SOMESPECIALSTRING
statments
...
SOMESPECIALSTRING
這裏要求在須要鍵盤輸入的命令後,直接加上 <<符號,而後跟上一個特別的字符串,在該串後按順序輸入原本應該由鍵盤輸入的全部字符,在全部須要輸入的字符都結束後,重複一遍前面 <<符號後的「特別的字符串」即表示該輸入到此結束。
在 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 中了。
用 bash -x bash-script 命令,能夠查看一個出錯的 BASH 腳本到底錯在什麼地方,能夠幫助程序員找出腳本中的錯誤。
另外用 trap 語句能夠在 BASH 腳本出錯退出時打印出一些變量的值,以供程序員檢查。trap 語句必須做爲繼 "#!/bin/bash" 後的第一句非註釋代碼,通常 trap 命令被寫做: trap 'message $checkvar1 $checkvar2' EXIT 。
使用 bash -version 命令能夠看出當前你正在使用的 BASH 是什麼版本,通常版本號爲1.14或其餘版本。而如今機器上通常還安裝了一個版本號爲 2.0 的 BASH 。該 BASH 也在 /bin 目錄下。BASH2 提供了一些新功能,有興趣的同敘能夠本身去看相關資料,或直接 man bash2 便可。