shell函數傳回值,常用的兩種方式:return,echo
1) return 語句shell函數的傳回值,可以和其他語言的傳回值一樣,通過return語句傳回。
示例1:
“numeric argument required”錯誤:
shell 函數傳回值隻能是整形數值,一般是用來表示函數執行成功與否的,0表示成功,其他值表示失敗。因而用函數傳回值來傳回函數執行結果是不合适的。如果要硬生生地return某個計算結果,比如一個字元串,往往會得到錯誤提示:“numeric
argument required”。
如果一定要讓函數傳回一個或多個值,可以定義全局變量,函數将計算結果賦給全局變量,然後腳本中其他地方通過通路全局變量,就可以獲得那個函數“傳回”的一個或多個執行結果了。
重新修改下示例2:
示例3:
說明:先定義了一個函數mytest,根據它輸入的參數是否為1來return 100或者return 200。
a、如何調用shell函數(執行函數):函數名 參數
b、擷取shell函數結果:$?
c、另外,可以直接用函數的傳回值用作if的判斷。(示例3)
注意:return隻能用來傳回整數值,且和c的差別是傳回為正确,其他的值為錯誤。
2) argv全局變量
這種就類似于c語言中的全局變量(或環境變量)。
示例:
結果:
mytest2
args 1
return 0
g_var=1
函數mytest2通過修改全局變量(g_var)的值,來傳回結果。
注: 以上兩個方法失效的時候,以上介紹的這兩種方法在一般情況下都是好使的,但也有例外。
其中,test.txt 檔案中的内容如下:
456:kkk
123:yxb
123:test
yxb
mytest3 here
1
g_var=0
mytest4 here
g_var=
可以看到mytest3在return了以後其實沒有直接傳回,而是執行了循環體後的語句,同時看到mytest4中也是一樣,同時,在mytest4中,對全局變量的修改也無濟于事,全局變量的值根本就沒有改變。這個是什麼原因那?
筆者認為,之是以return語句沒有直接傳回,是因為return語句是在管道中執行的,管道其實是另一個子程序,而return隻是從子程序中傳回而已,隻是while語句結束了。而函數體之後的語句會繼續執行。
同理,全局變量在子程序中進行了修改,但是子程序的修改沒有辦法反應到父程序中,全局變量隻是作為一個環境變量傳入子程序,子程序修改自己的環境變量,不會影響到父程序。
是以在寫shell函數的時候,用到管道(cmd &背景程序也一樣)的時候一定要清楚此刻是從什麼地方傳回。
3) echo 傳回值
其實在shell中,函數的傳回值有一個非常安全的傳回方式,即通過輸出到标準輸出傳回。因為子程序會繼承父程序的标準輸出,是以,子程序的輸出也就直接反應到父程序。
示例4:
test 1
arg1 = 1
test 0
arg1 = 0
test 2
arg1 = 2
怎麼?有沒有搞錯,這兩個函數幾乎沒什麼差別,對,它幾乎就是return和echo不一樣,但是有一點一定要注意,不能向标準輸出一些不是結果的東西(也就是說,不能随便echo一些不需要的資訊),比如調試資訊,這些資訊可以重定向到一個檔案中解決,特别要注意的是,腳本中用到其它類似grep這樣的指令的時候,一定要記得1>/dev/null 2>&1來空這些輸出資訊輸出到空裝置,避免這些指令的輸出。
如上shell代碼段,執行結果是
return: 100
dd: test here
如果 把#echo excute over注釋去掉,執行結果是
excute over
return: 0
也就是說:$?是接收上一條函數的執行結果,在例一中,$?接收test函數的執行結果,執行結果其實就是其傳回值,就是return 傳出來的數值(return隻能是數字,不能是字元串之類的),如果函數中沒有顯式調用return傳回出來狀态,那麼系統會使用函數中最後一條shell指令的執行結果作為傳回值,如果函數test()最後一條指令調用其他函數,如:test1(),那麼test的傳回值就是test1的傳回值。
dd是作為一個變量接收函數的标準輸出,比如echo産生出來的資訊,不包括報錯之類的資訊,如上如果函數test中調用test1,那麼test和test1中的标準輸出都會指派給變量dd.
注意:dd=`test` , ` `符号不是單引号‘’,而是esc下面的那個符号,該句話的意思是執行函數test結果指派給dd,如果是單引号的話這句話是把字元串“test”指派給變量dd,注意,這句話的執行結果不是指派是否成功,而是函數的執行狀态
總結:是以,可以總結一下函數傳回值擷取的方法:
1)用變量接收函數傳回值,函數用echo等标準輸出将要傳回的東西列印出來。
2)用$?來接收函數的執行狀态,但是$?要緊跟在函數調用處的後面。