shell调用函数返回值深入分析
编写shell脚本过程中,我们经常会自定义一些函数,并根据函数的返回值不同来执行相应的流程,那么我们如何来获取函数的返回值呢?
首先shell中调用函数有两种方式:
第一种:value=`function_name [arg1 arg2 ......]`
或
第二种:function_name [arg1 arg2 ......]
echo $?
这两种有什么区别呢?
举个例子来说:
[root@zejin240 ~]# cat test.sh
#!/bin/sh
function aaa()
{
if [ "$1" -eq "1" ];then
return 0
elif [ "$1" -eq "2" ];then
exit 1
elif [ "$1" -eq "3" ];then
echo "3"
echo "31" >&2 #no space after >
echo "32"
elif [ "$1" -eq "4" ];then
a wrong commend
fi
}
echo "begin"
value1=`aaa 1`
echo $?
echo "value1=$value1"
echo '---------------'
value2=`aaa 2`
echo $?
echo "value2=$value2"
echo '---------------'
value3=`aaa 3`
echo $?
echo "value3=$value3"
echo '---------------'
value4=`aaa 4`
echo $?
echo "value4=$value4"
echo "end"
输出如下:
[root@zejin240 ~]# sh test.sh
begin
0
value1=
---------------
1
value2=
---------------
31
0
value3=3
32
---------------
test.sh: line 14: a: command not found
127
value4=
---------------
end
可以看到,value1 value2 value3 value4的值取决于函数中的标准输出的值,即函数体中的echo内容,注意value3的值,因为我们将31的输出定向到了标准错误输出,而不会出现在value3的值中。
而我们用$?获得的函数执行状态值时,可以看到,没有加return和exit的直接返回它上一条命令执行的状态值,所以echo "32"可以正常返回,所以$?的值为0,而a wrong command 由于不能正常执行,而返回了一个非0的整数。
好了,那么让我们来看下面一个命令:
[root@zejin240 ~]# /opt/lamp/mysql/bin/mysql -h192.168.1.240 -uDBA -p123 -ss -e 'show variables like "version"'
Warning: Using a password on the command line interface can be insecure.
version 5.6.16-log
输出中出现了一个让人有点讨厌的警告,但是呢,当我把命令的值赋与变量aa时
[root@zejin240 ~]# aa=`/opt/lamp/mysql/bin/mysql -h192.168.1.240 -uDBA -p123 -ss -e 'show variables like "version"'`
Warning: Using a password on the command line interface can be insecure.
[root@zejin240 ~]# echo "$aa"
version 5.6.16-log
你会发现,变量aa的值并没有包含那个警告信息。为什么呢,看了上面的分析能否知道原因呢
原因是这样子的,虽然在界面显示出来了警告信息,但这个警告信息是被重定向到标准错误输出的,并不会被赋值给变量aa,我是怎么知道的呢
我们可以这样做个实验:
[root@zejin240 ~]# aa=`/opt/lamp/mysql/bin/mysql -h192.168.1.240 -uDBA -p123 -ss -e 'show variables like "version"' 2>&1 `
[root@zejin240 ~]# echo "$aa"
Warning: Using a password on the command line interface can be insecure.
version 5.6.16-log
我将标准错误输出重定位到标准输入,你看,这变量aa的值就包含警告信息了吧。
当然我们也可以将标准输出重定向到空设备中:
[root@zejin240 ~]# aa=`/opt/lamp/mysql/bin/mysql -h192.168.1.240 -uDBA -p123 -ss -e 'show variables like "version"' >/dev/null 2>&1`
[root@zejin240 ~]# echo "$aa"
你看,aa的值就为空了吧。(即使是定向到某个文件中输出也为空)
总结:调用函数返回值时,不管方法一还是方法二,echo $?的值是一样的。而方法一中,即value=`function_name [arg1 arg2 ......]`,value的取值取决于调用命令的标准输出,没有标准输出即为空,我们一般用return值来返回函数的最后执行状态,(一般不用exit来返回,因为它直接调用时会直接结束整个shell,当然如果有特殊目的可以这样做),用echo来输出给到变量的值。