歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux編程 >> SHELL編程 >> shell編程其實真的很簡單(五)

shell編程其實真的很簡單(五)

日期:2017/3/2 17:31:21   编辑:SHELL編程

命令行參數處理

根據參數位置獲取參數

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


Copyright © Linux教程網 All Rights Reserved