1. 程式人生 > >shell 指令碼中獲取命令的輸出

shell 指令碼中獲取命令的輸出

這個主要介紹的方法是獲取命令的輸出內容,而不是命令執行成功與否的返回值。

通常情況下,在shell指令碼中需要獲取命令的輸出內容,然後根據輸出內容判斷下一步的執行操作。

比較常用的一種方式就是, 匹配命令輸出的內容中是否存在某些關鍵字,選擇執行的不同動作。

比較常用的一種方式就是採用反向單引號的方式 --  儲存結果的變數名=`需要執行的linux命令`


這種方式在使用時,有些細節的地方需要注意。 先用幾個例子來說明一下。

比如在CentOS7環境中,使用rpm -qa命令查詢某些rpm包是否安裝,沒有安裝的話進行安裝操作。

舉個簡單的例子來說:

#!/bin/bash

check_results=`rpm -qa | grep "zlib"`
echo "command(rpm -qa) results are: $check_results"
if [[ $check_results =~ "zlib" ]] 
then 
    echo "package zlib has already installed. "
else 
    echo "This is going to install package zlib"
 fi

儲存為test.sh檔案,然後執行

$bash test.sh 

結果為:
command(rpm -qa) results are: zlib-1.2.7-13.el7.x86_64
package zlib has already installed. 

這個指令碼基本上是可以工作的。

那麼,我們同樣使用類似的方式來檢查iscsi-initiator軟體包是否安裝。 與上一個命令不同的是,這個命令是否安裝不能通過rpm -qa命令獲取。

我們採取另一種方式

#!/bin/bash

check_results=`iscsiadm --version | grep iscsiadm`
echo "check command(iscsiadm) available results are: $check_results"
if [[ $check_results =~ "iscsiadm" ]] 
then 
    echo "command iscsiadm could be used already."
else     
    echo "command iscsiadm can't be used. Install it"
    rpm -ivh iscsi-initiator-utils-6.2.0.873-29.el7.x86_64.rpm
fi

執行的結果為:

$ bash test.sh
check command(iscsiadm) available results are: iscsiadm version 6.2.0.873-28
command iscsiadm could be used already.

這時候看起來指令碼是工作正常的,顯示iscsiadm已經可用。那麼假如一開始的時候iscsiadm命令不可用呢?

我們可以將上面的check_results=`iscsiadm --version | grep iscsiadm`改成為check_results=`iscsiadmm --version | grep iscsiadm`

這樣我們故意將命令寫錯,模擬命令沒有安裝的情況下指令碼的執行。

修過後執行結果如下

$ bash test.sh
test.sh: line 2: iscsiadmm: command not found
check command(iscsiadm) available results are: 
command iscsiadm can't be used. Install it

看起來好像也是工作正常的,但是為什麼check_results的內容為空呢? 難道不應該將“command not found: iscsiadmm”內容賦值給check_results,然後列印顯示出來嗎?

我們將iscsiadmm --version | grep iscsiadm單獨在命令列中執行,也是有輸出的啊,為什麼不能賦值成功呢?

這個時候如果我們在命令執行完畢之後,執行命令echo$? 這個時候得到的值為127(Centos7系統)。在命令列中單獨執行scsiadm --version | grep iscsiadm命令之後,執行echo $?得到的值為0(Centos7系統)

從這裡可以看出在使用··(2個反向單引號)的方式獲取執行結果時需要保證單引號內的命令是可以執行成功的。


就算是這樣就能保證我們可以獲取到想要的內容嗎? 不一定,再來看個例子。

#!/bin/bash

check_results=`java -version`
echo "check java version results are: $check_results"
if [[ $check_results =~ "1.8." ]] 
then 
    echo "java version is 1.8, it seems not need to install java again."
else     
    echo "It is going to install jdk 1.8 version"
fi

執行結果為:

$ bash test.sh                                                                                 
java version "1.8.0_73"
Java(TM) SE Runtime Environment (build 1.8.0_73-b02)
Java HotSpot(TM) 64-Bit Server VM (build 25.73-b02, mixed mode)
check java version results are: 
It is going to install jdk 1.8 version

怎麼回事?命令jdk已經安裝了,為什麼還是沒有匹配到呢?根據第二個例子,我們手動執行命令 java -version 然後執行命令$? 得到的結果為0, 說明命令是執行成功的。怎麼還是沒有獲取到命令的輸出呢?

這個現象出現的原因是有可能命令的執行結果被重定向了。

試著將check_results=`java -version` 改成check_results=`java -version 1> /dev/stdout` 和 check_results=`java -version 2> /dev/sdtout` 看看輸出是否有變化

為了防止這種情況的發生,最終我們將上面改成check_results=`java -version 2>&1` 這樣得到了想要的結果。


使用  儲存結果的變數名=`需要執行的linux命令` 這種方式來獲取命令的輸出時,注意的情況總結如下:

1)保證反單引號內的命令執行時成功的,也就是所命令執行後$?的輸出必須是0,否則獲取不到命令的輸出

2)即便是命令的返回值是0,也需要保證結果是通過標準輸出來輸出的,而不是標準錯誤輸出,否則需要重定向

因此我們推薦使用  儲存結果的變數名=`需要執行的linux命令 2>&1 `的方式來獲取命令的執行結果。

感興趣的朋友可以試下第二個例子中改成  check_results=`iscsiadmm --version 2>&1`的結果。


此外還有一種獲取命令執行返回值的方式 變數名=$(需要執行的命令) 對於這種方式沒有進行測試,所以不再此討論。

對於上面提到的獲取命令執行輸出的情況,和獲取函式執行結果的方式並不同,請在使用中進行注意。