Linux Shell函数返回值

2019-09-23 09:16:18王振洲

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来避免这些命令的输出。