yxb
mytest3 here
1
yxb
g_var=0
mytest4 here
1
g_var=
可以看到mytest3在return了以后其实没有直接返回,而是执行了循环体后的语句,同时看到mytest4中也是一样,同时,在mytest4中,对全局变量的修改也无济于事,全局变量的值根本就没有改变。这个是什么原因那?
笔者认为,之所以return语句没有直接返回,是因为return语句是在管道中执行的,管道其实是另一个子进程,而return只是从子进程中返回而已,只是while语句结束了。而函数体之后的语句会继续执行。
同理,全局变量在子进程中进行了修改,但是子进程的修改没有办法反应到父进程中,全局变量只是作为一个环境变量传入子进程,子进程修改自己的环境变量,不会影响到父进程。
因此在写shell函数的时候,用到管道(cmd &后台进程也一样)的时候一定要清楚此刻是从什么地方返回。
3) echo 返回值
其实在shell中,函数的返回值有一个非常安全的返回方式,即通过输出到标准输出返回。因为子进程会继承父进程的标准输出,因此,子进程的输出也就直接反应到父进程。因此不存在上面提到的由于管道导致返回值失效的情况。
在外边只需要获取函数的返回值即可。
示例:
#!/bin/bash
##############################################
# Author : IT-Homer
# Date : 2012-09-06
# Blog : http://blog.csdn.net/sunboy_2050
##############################################
function mytest5()
{
grep "123" test.txt | awk -F: '{print $2}' | while read line; do
if [ $line = "yxb" ]; then
echo "0" # value returned first by this function
return 0
fi
done
return 1
}
echo '$? = '"$?"
result=$(mytest5)
echo "result = $result"
echo
if [ -z $result ] # string is null
then
echo "no yxb. result is empyt"
else
echo "have yxb, result is $result"
fi
结果:
$? = 0
result = 0
have yxb, result is 0
这个方式虽然好使,但是有一点一定要注意,不能向标准输出一些不是结果的东西,比如调试信息,这些信息可以重定向到一个文件中解决,特别要注意的是,用到比如grep这样的命令的时候,一定要记得1>/dev/null 2>&1来避免这些命令的输出。










