bash shell可根據參數位置獲取參數。通過 $1 到 $9 獲取第1到第9個的命令行參數。$0為shell名。如果參數超過9個,那麼就只能通過${}來獲取了, 例如獲取第10個參數,那麼可以寫為${10}。
示例一:
#!/bin/bash
#testinput.sh
echo "file name: $0"
echo "base file name: $(basename $0)"
echo "param1: $1"
echo "param2: ${2}"
運行上面的的shell
./testinput.sh 12 34
最終得到的結果如下:
file name: ./testinput4.sh
base file name: testinput4.sh
param1: 12
param2: 34
成功的得到文件名和命令行輸入的參數(命令行參數以空格分隔,如果參數包含了空格,那麼久必須添加引號了)
$0默認會獲取到當前shell文件的名稱,但是,它也包含(./),如果你以完整路徑運行,那麼這還會包含目錄名。因此,上面通過basename命令來獲取單純的文件名$(basename $0)。
試想一下,假如我們寫的shell的這個參數很多,那如果像上面那樣一個一個去獲取參數,那豈不是要寫瘋!下面就來看看如何解決這種情況。
既然bash shell通過位置可獲取參數,那意味著如果我們知道參數的總個數就可以通過循環依次獲取參數。那麼如何獲取參數總個數呢?
在bash shell中通過 $# 可獲取參數總數。
示例:(循環獲取參數)
#!/bin/bash
for (( index=0; index <= $#; index++ ))
do
echo ${!index}
done
以上示例,我們通過 $# 獲取總參數個數。然後通過循環獲取每個位置的參數。注意: 按照正常的理解,上面的 ${!index} 應該是 ${$index}才對, 對吧? 但是,由於${}內不能再寫$符號,bash shell在這個地方是用了!符號,所以以上才寫為了${!index}。
在bash shell中還可以通過 $* 和 $@ 來獲取所有參數。但是這兩者之間有著很大的區別:
$* 會將命令行上提供的所有參數當作一個單詞保存, 我們得到的值也就相當於是個字符串整體。
$@ 會將命令行上提供的所有參數當作同一字符串中的多個獨立的單詞。
可能文字看起來描述的不太清楚,那麼還是通過示例來看二者的區別吧:
#!/bin/bash
#testinput.sh
var1=$*
var2=$@
echo "var1: $var1"
echo "var2: $var2"
countvar1=1
countvar2=1
for param in "$*"
do
echo "first loop param$countvar1: $param"
countvar1=$[ $countvar1 + 1 ]
done
echo "countvar1: $countvar1"
for param in "$@"
do
echo "second param$countvar2: $param"
countvar2=$[ $countvar2 + 1 ]
done
echo "countvar2: $countvar2"
執行上面的示例:
./testinput.sh 12 34 56 78
上面示例的輸出結果為:
var1: 12 34 56 78
var2: 12 34 56 78
param1: 12 34 56 78
countvar1: 2
param1: 12
param2: 34
param3: 56
param4: 78
countvar2: 5
通過上面的結果可見,直接輸出看起來二者結果一樣,但是通過for循環就可看出二者的區別了。上一篇文章我們講到for循環會通過IFS定義的值進行分割,因此默認情況下,如果我們上面在for循環處不加引號,那麼根據IFS中所定義的空格分割,最終也會導致看不出二者區別。
有時候,我們在shell執行過程中獲取用戶的輸入,以此與用戶進行交互。這是通過read命令來實現的。下面就來看看其用法:
示例一:
#!/bin/bash
echo -n "yes or no(y/n)?"
read choice
echo "your choice: $choice"
運行以上示例,首先會輸出”yes or no(y/n)?“, 然後會等待用戶輸入(-n參數表示不換行,因此會在本行等待用戶輸入),當用戶輸入後,會把用戶輸入的值賦值給choice變量, 然後最終輸出 “your choice: (你輸入的內容)”。
事實上,我們可以不指定read後面的變量名,如果我們不指定, read命令會將它收到的任何數據都放進特殊環境變量REPLY中。如下:
示例二:
#!/bin/bash
echo -n "yes or no(y/n)?"
read
echo "your choice: $REPLY"
以上示例與示例一是等價的。
有時候,我們需要用戶輸入多個參數,當然,shell是支持一次接受多個參數輸入的。
示例三:
#!/bin/bash
read -p "what's your name?" first last
echo first: $first
echo last: $last
以上示例首先輸出“what's your name?”, 然後在本行等待用戶輸入(此處用read -p實現以上示例的echo -n + read命令的不換行效果),輸入的參數以空格分隔,shell會把輸入的值依次賦值給first和last兩個變量。如果輸入的值過多,假如我輸入了3個值,那麼shell會把剩下的值都賦值給最後一個變量(即第二三兩個的值都會賦值給last變量)。
細想一下,有個問題,假如用戶一直不輸入,怎麼辦?一直等待?
我們可以通過read -t 來指定超時時間(單位為秒),如果用戶在指定時間內沒輸入,那麼read命令就會返回一個非0的狀態碼。
示例四:
#/bin/bash
if read -t 5 -p "Please enter your name: " name
then
echo "Hello $name"
else
echo "Sorry, timeout! "
fi
運行以上示例,如果超過5秒沒輸入,那麼就會執行else裡面的。
本篇簡單的介紹了shell的輸入參數以及接收用戶輸入。大家可以舉一反三,結合之前所學的基礎知識,可以寫一些小的腳本應用了。
我的獨立博客: javafan.cn