Linux Shell编程中也会使用到函数,函数可以把大的命令集合分解成若干较小的任务,也避免重复编写相同的代码。在Shell中,所有的函数定义都是平行的,即不允许在函数体内再定义其他的函数,但允许函数之间相互调用。函数又涉及到很多基本使用:函数参数调用、函数返回值、局部变量及全局变量、函数间的相互调用和函数递归调用。
(1)函数是一串命令的集合,如果脚本中有重复代码时可以考虑使用函数,在shell中函数的返回值只能为退出状态0或1。应注意函数名在脚本中的唯一。可以在函数名前加上fuction关键字也可省略。函数体中的命令集合必须含有至少一条命令(函数不允许空命令,与C不同)。
(2)脚本遇到"hello(){"结构时,知道定义了一个名为hello的函数,而且它会记住hello代表的函数,并执行函数体中的命令,直到出现"}"字符结束,当执行到单独的行hello时,shell就知道应该去执行刚才定义的函数,执行完毕返回继续执行其他的命令或函数。在shell中不需要申明就可直接定义函数,但在调用函数前需对它进行定义。如下是shell脚本中循环调用函数的例子:
#function2.sh #!/bin/bash output() { for(( num1 = 1; num1 <= 5; num1++ )) do echo -n "$num1 " done } let "num2=1" while [ "$num2" -le 5 ] do output echo "" let "num2=num2 + 1" done
#显示当前目录下文件和目录数 #function3.sh #!/bin/bash directory() { let "filenum = 0" let "dirnum = 0" ls echo "" #echo换行 for file in $( ls ) #for循环判断当前子目录和文件 do if [ -d $file ] #判断为子目录 then let "dirnum = dirnum + 1" else #判断为文件 let "filenum = filenum + 1" fi done echo "The number of directory is $dirnum" echo "The number of file is $filenum" } directory #在脚本中调用函数
shell中,向函数传递的参数仍然是以位置参数的方式来传递的,而不能传递数组等其他形式变量(与c不同)。利用函数参数的传递实现两数的四则运算。
#用于实现两数加、减、乘和除四则运算 #!/bin/bash count() { if [ $# -ne 3 ] #3个参数,一个运算符两个数值 then echo "The number of arguments is not 3! " fi let "s = 0" case $2 in +) #加法 let "s = $1 + $3" echo "$1 + $3 = $s";; -) #减法 let "s = $1 - $3" echo "$1 - $3 = $s";; \*) #乘法 let "s = $1 * $3" echo "$1 * $3 = $s";; \/) #除法-取余,浮点数运算使用bc let "s = $1 / $3" echo "$1 / $3 = $s";; *) #其他 echo "What you input is wrong!";; esac } echo "Please type your word: ( e.g. 1 + 1 )" read a b c count $a $b $c
有时需要函数执行完成后返回特定的值来完成脚本的后续操作。函数通过return返回退出状态,0表示true无错误,非0表示false有错误(与C不同)。
#根据用户输入显示星期 #!/bin/bash show_week() { echo -n "What you input is: " echo "$1" case $1 in 0) echo "Today is Sunday. " return 0;; 1) echo "Today is Monday. " return 0;; 2) echo "Today is Tuesday. " return 0;; 3) echo "Today is Wednesday. " return 0;; 4) echo "Today is Thursday. " return 0;; 5) echo "Today is Friday. " return 0;; 6) echo "Today is Saturday. " return 0;; *) return 1;; esac } #if show_week "$1" #返回0表示函数输入的命令行参数是正确的 #也可以通过$?来获取函数执行的返回值 show_week "$1" if [ $? -eq 0 ] then echo "What you input is right! " else echo "What you input is wrong! " fi exit 0
shell脚本中可以同时放置多个函数,函数之间允许相互调用,而且允许一个函数调用多个函数。
#用于显示一个不多于5位的正整数的位数,并按顺序显示各个数位的值 #!/bin/bash count_of_int() { if [ $1 -gt 9999 ] then let "place=5" #5位数 elif [ $1 -gt 999 ] then let "place=4" elif [ $1 -gt 99 ] then let "place=3" elif [ $1 -gt 9 ] then let "place=2" else let "place=1" #1位数 fi echo "The place of the $1 is $place." #整数的位数 } num_of_int() { let "ten_thousand = $1/10000" #整数的数位分解值 let "thousand =$1/1000%10" let "hundred = $1/100%10" let "ten = $1%100/10" let "indiv = $1%10" if [ $ten_thousand -ne 0 ] then echo "$ten_thousand $thousand $hundred $ten $indiv" elif [ $thousand -ne 0 ] then echo "$thousand $hundred $ten $indiv" elif [ $hundred -ne 0 ] then echo "$hundred $ten $indiv" elif [ $ten -ne 0 ] then echo "$ten $indiv" else echo "$indiv" fi } show() { echo "Please input the number(1-99999): " read num count_of_int $num num_of_int $num } show执行脚本,输入整数2014,该数是四位数,千位是2,百位是0,十位是1,个位是4。
shell中,可以通过local关键字来申明局部变量,局部变量将局限在函数范围内。函数可调用函数外的全局变量,若一个局部变量和一个全局变量的名字相同,则在函数中局部变量会覆盖掉全局变量。
#!/bin/bash text="global variable" use_local_var_fun() { local text="local variable" #local声明为局部变量,否则会修改全局text的值 echo "in function------"$text #暂时覆盖掉全局变量的值 } use_local_var_fun echo "Out of function------"$text #text的值为开始定义的值 exit 0
(1)shell中执行递归函数可以直接或间接地反复调用其自身,每调用一层就进入新的一层,主调函数又是被调函数。
(2)使用局部变量进行递归实现阶乘运算,也可采用递推法实现。
#阶乘运算,当n=0时 0!=1,当n>=1时n!=n*(n-1)! #!/bin/bash fact () { local num=$1 #函数需将先前的局部值进行逐个还原,故设置成局部的变量 echo $num if [ "$num" -eq 0 ] then factorial=1 else let "decnum=num-1" fact $decnum let "factorial=$num * $?" fi return $factorial } fact $1 echo "Factorial of $1 is $?" exit 0(3)递归实现汉诺塔的问题(不使用局部变量的递归)
#汉诺塔算法 #!/bin/bash move=0 dohanoi() { if [ $1 -eq 0 ] then echo "" else dohanoi "$(($1-1))" $2 $4 $3 echo "move $2 ----> $3" let "move=move+1" dohanoi "$(($1-1))" $4 $3 $2 fi if [ $# -eq 1 ] then if [ "$(( $1 > 1 ))" -eq 1 ] then dohanoi $1 A C B echo "Total moves = $move" else echo "The number of disk which you input is illegal! " fi fi } echo "Please input the num of disk:" read num dohanoi $num 'A' 'B' 'C'
(1)函数间的相会调用增加了shell编程的灵活性和代码的可重用性,对脚本语言来说很是实用。
(2)函数的递归调用应进一步理解,阶乘和汉诺塔的实现可逐步分析。
(3)通过函数可以封装自己的函数库,减少以后开发的难度并使用代码的可重复性