完整版 Linux技巧:使用 bash function 命令自定义函数( 二 )


执行某个函数后,可以使用 $? 表达式来获取函数的 return 返回值,但是要注意下面的一种情况:
var=$(foo)if [ "$?" == "0" ]; thenecho successfi此时,不要在 var=$(foo) 和 if [ "$?" == "0" ]; then 之间添加任何语句 。
否则,$? 获取到将不是 $(foo) 的 return 值,判断就有问题,特别是不要添加 echo 调试语句 。
换句话来说,这种先执行一个语句,再判断 $? 的方法不是很可靠,会受到各种影响,要特别注意代码语句的顺序 。
使用函数名作为函数指针在 bash 中,可以通过如下的方式来达到类似C语言函数指针的效果 。
假设有一个 test.sh 脚本,内容如下:
#!/bin/bash# 注意$1后面有一个分号';', 少这个分号会报错echo_a() { echo aaaa $1; }echo_b() { echo bbbb $1; }if [ "$1" == "-a" ]; then# 这里的 echo_a 没有加双引号echo_common=echo_aelif [ "$1" == "-b" ]; then# 上面的echo_a没加双引号, 这里加了.# 实际上, 可加可不加, 都可以正确执行.echo_common="echo_b"elseecho ERROR; exit 1fi${echo_common} common这个脚本通过 echo_common 变量来保存函数名,相当于是函数指针,再通过 ${echo_common} 来调用它保存的函数 。
在 bash shell 中执行 ./test.sh -a 命令,会输出 "aaaa common";执行 ./test.sh -b 命令,会输出 "bbbb common" 。
函数内执行cd命令的影响在函数里面执行 cd 命令,切换到某个目录后,函数退出时,当前工作目录还是会保持在那个目录,而不会自动恢复为原先的工作目录,需要手动执行 cd - 命令再切换回去 。
假设有一个 testcd.sh 脚本,里面的内容如下:
#!/bin/bashecho "now, the pwd is: $(pwd)"cd_to_root() { cd /usr/; }cd_to_rootecho "after execute the cd_to_root, pwd is: $(pwd)"这个函数先打印出执行脚本时工作目录路径,然后执行自定义的 cd_to_root 函数,在函数内部切换工作目录到 "/usr/",最后在 cd_to_root 函数外面打印工作目录路径 。
执行 ./testcd.sh 脚本,会输出下面的内容:
[~/sample]$ ./testcd.shnow, the pwd is: /home/sampleafter execute the cd_to_root, pwd is: /usr[~/sample]$ pwd/home/sample可以看到,如果在函数里面执行过 cd 命令,函数退出后,当前工作目录还是 cd 后的目录 。
但是脚本执行结束后,当前 shell 的工作目录还是之前的工作目录,不是脚本里面 cd 后的目录 。
在每个 shell 下面,当前工作目录 (working directory ) 是全局的状态,一旦改变,在整个 shell 里面都会改变 。
而 bash 执行脚本时,是启动一个新的子 shell 来执行,所以脚本内部执行 cd 命令,会影响运行这个脚本的子 shell 的工作目录,但不影响原先父 shell 的工作目录 。
声明函数内的变量为局部变量在 bash 中,没有使用 local 命令来声明的变量都是全局变量,在函数内部定义的变量也是全局变量 。
如果没有注意到这一点,在函数内操作变量可能会影响到的外部同名变量,造成不预期的结果 。
为了避免对外部同名变量造成影响,函数内的变量最好声明为局部变量,使用 local 命令来声明 。
【完整版 Linux技巧:使用 bash function 命令自定义函数】查看 man bash 里面对 local 命令的说明如下:

local [option] [name[=value] ...]
For each argument, a local variable named name is created, and assigned value. The option can be any of the options accepted by declare.
When local is used within a function, it causes the variable name to have a visible scope restricted to that function and its children. With no operands, local writes a list of local variables to the standard output.
It is an error to use local when not within a function.
The return status is 0 unless local is used outside a function, an invalid name is supplied, or name is a readonly variable.
即,在函数内,使用 local 命令声明的变量是局部变量,这些变量在函数外不可见 。
在函数外,不能使用 local 命令声明变量,否则会报错 。
注意:如上面说明,local 命令本身会返回一个值,正常的返回值是 0 。
假设有个 testlocal.sh 脚本,内容如下:
#!/bin/bashfoo() { return 1; }bar() {ret=$(foo); echo first: $?local var=$(foo); echo second: $?}foobar() { return 0; }barlocal out=$(foobar); echo third: $?则执行 ./testlocal.sh 脚本,会输出:
first: 1second: 0./testlocal.sh: 第 x 行:local: 只能在函数中使用third: 1可以看到,ret=$(foo); 语句没有使用 local 命令来定义 ret 变量,执行 foo 函数,该函数的返回值是 1,所以得到的 $? 是 1 。


推荐阅读