不灭的焱

革命尚未成功,同志仍须努力下载JDK17

作者:Albert.Wen  添加时间:2013-09-05 17:09:24  修改时间:2024-04-29 17:12:08  分类:Linux基础  编辑

shell函数返回值一般有3种方式:

1、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的区别是返回为正确,其他的值为错误。

2、全局变量或者环境变量

这种就类似于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通过修改全局变量的值来返回结果。

3、以上两个方法失效的时候

以上介绍的这两种方法在一般情况下都是好使的,但也有例外。

比如:

#!/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

输出如下:

yxb@yxb-laptop:~/文档/个人文档/shell函数返回值$ ./no_function.sh
yxb
mytest3 here
1
yxb
g_var=0
mytest4 here
g_var=

可以看到mytest3return了以后其实没有直接返回,而是执行了循环体后的语句,同时看到mytest4中也是一样,同时,在mytest4中,对全局变量的修改也无济于事,全局变量的值根本就没有改变。

这个是什么原因那?

笔者认为,之所以return语句没有直接返回,是因为return语句是在管道中执行的,管道其实是另一个子进程,而return只是从子进程中返回而已,只是while语句结束了,而函数体之后的语句会继续执行。

同理,全局变量在子进程中进行了修改,但是子进程的修改没有办法反应到父进程中,全局变量只是作为一个环境变量传入子进程,子进程修改自己的环境变量,不会影响到父进程。

因此在写shell函数的时候,用到管道的时候一定要清楚,此刻是从什么地方返回。

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

echo输出另一技巧:用函数的返回值作为另外一个函数的参数

#!/bin/bash 
Dir=/cygdrive/d/server/ebin 
function display() {
	files=`ls $Dir`
	echo $files 
}
echo `display` 
 
function filetype() {
    echo `file $Dir/$1`  #输出待检测文件的类型 
} 
 
for file in `display`    #调用display函数,对其返回值做遍历 
do 
    filetype $file       #检测文件类型并输出 
done 

 

小总结:

$? 来获取函数的 return值,用 $(函数名) 来获取函数的 echo值