天天看點

Shell函數傳回值

shell函數傳回值一般有3種方式:

return語句

shell函數的傳回值可以和其他語言的傳回值一樣,通過return語句傳回。

比如:

#!/bin/bash -

function mytest()

{

echo "mytest function"

echo "argv[1] = $1"

if [ $1 = "1" ] ;then

return 1

else

return 0

fi

}

echo "mytest 1"

mytest 1

echo $?

echo "mytest 0"

mytest 0

echo $?

if mytest 1 ; then

echo "mytest 1"

fi

if mytest 0 ;then

echo "mytest 0"

fi

echo "end"
           

先定義了一個函數,mytest,它根據輸入的參數是否為1來return 1或者return 0.

擷取函數的傳回值通過調用函數,或者最後執行的值獲得。

另外,可以直接用函數的傳回值用作if的判斷。

注意:return隻能用來傳回整數值,且和c的差別是傳回為正确,其他的值為錯誤。

全局變量或者環境變量

這種就類似于c中的全局變量。

#!/bin/bash -

g_var=

function mytest2()

{

echo "mytest2"

echo "args $1"

g_var=$1

return 0

}

mytest2 1

echo "g_var=$g_var"
           

函數mytest2通過修改全局變量的值來傳回結果。

以上兩個方法失效的時候

以上介紹的這兩種方法在一般情況下都是好使的,但也有例外。

比如:

#!/bin/bash -

function mytest3()

{

grep "123" test.txt | awk -F: '{print $2}' | while read line ;do

echo "$line"

if [ $line = "yxb" ]

then

return 0

fi

done

echo "mytest3 here "

return 1

}

g_var=

function mytest4()

{

grep "123" test.txt | awk -F: '{print $2}' | while read line ;do

echo "$line"

if [ $line = "yxb" ]

then

g_var=0

echo "g_var=0"

return 0

fi

done

echo "mytest4 here "

return 1

}

mytest3

echo $?

mytest4

echo "g_var=$g_var"

test.txt中的内容如下:

456:kkk

123:yxb

123:test

輸出如下:

[email protected]:~/文檔/個人文檔/shell函數傳回值$ ./no_function.sh

yxb

mytest3 here

1

yxb

g_var=0

mytest4 here

g_var=
           

可以看到mytest3在return了以後其實沒有直接傳回,而是執行了循環體後的語句,同時看到mytest4中也是一樣,同時,在mytest4中,對全局變量的修改也無濟于事,全局變量的值根本就沒有改變。

這個是什麼原因那?

筆者認為,之是以return語句沒有直接傳回,是因為return語句是在管道中執行的,管道其實是另一個子程序,而return隻是從子程序中傳回而已,隻是while語句結束了。而函數體之後的語句會繼續執行。

同理,全局變量在子程序中進行了修改,但是子程序的修改沒有辦法反應到父程序中,全局變量隻是作為一個環境變量傳入子程序,子程序修改自己的環境變量,不會影響到父程序。

是以在寫shell函數的時候,用到管道的時候一定要清除,此刻是從什麼地方傳回。

Echo傳回值

其實在shell中,函數的傳回值有一個非常安全的傳回方式,即通過輸出到标準輸出傳回。因為子程序會繼承父程序的标準輸出,是以,子程序的輸出也就直接反應到父程序。是以不存在上面提到的由于管道導緻傳回值失效的情況。

在外邊隻需要擷取函數的傳回值即可。

#!/bin/bash -

function mytest5()

{

grep "123" test.txt | awk -F: '{print $2}' | while read line ;do

if [ $line = "yxb" ]

then

echo "0"

return 0

fi

done

return 1

}

result=$(mytest5)

if [ -z $result ]

then

echo "no yxb. result is empyt"

else

echo "have yxb, result is $result"

fi

輸出如下:

have yxb, result is 0
           

這個方式雖然好使,但是有一點一定要注意,不能向标準輸出一些不是結果的東西,比如調試資訊,這些資訊可以重定向到一個檔案中解決,特别要注意的是,用到比如grep這樣的指令的時候,一定要記得1>/dev/null 2>&1來避免這些指令的輸出。

原文位址:http://my.oschina.net/biner/blog/28354