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