Shell脚本攻略--函数应用(返回值、传参、变量的作用范围、递归以及函数库)


一、概述

  • Shell函数也是我们经常使用的,因为有些命令序列是需要反复调用执行的,若每次使用同一命令就重复写一遍,就会导致代码量很大,行数特别多
  • 为解决该问题可以将命令序列按格式写一起,以便可以重复使用
  • 所以说,Shell函数的本质就是一段可以重复使用的脚本代码,这段代码被提前编写好了,放在了指定的位置,使用时直接调取即可

二、格式定义

  • [function]是可选的,表示该函数的功能,这个是可以省略掉的
  • 函数名后面跟一个(),里面是没有内容的
  • 而我们执行的命令序列放在{}里的
  • 若有些时候我们调用的函数很多,那么我们可以一次写好几个
  • 格式如下:
#格式一:

function 函数名 {
    
    
        命令序列
}


#格式二:

函数名() {
    
    
        命令序列
}


三、函数返回值

  • return表示退出函数并返回一个退出值,脚本中可以用$?变量显示该值
  • 使用原则:
    • 函数一结束就取返回值,因为$?变量只返回执行的最后一条命令的退出状态码
    • 退出状态码必须是0~255,超出时值将为除以256取余
  • 示例一:
[root@localhost sh]# vim xc1.sh

#!/bin/bash

function xcf {
    
    
        read -p "请输入:" a
        a=$[$a*2]
        return $a
}

xcf
echo $?

mark

  • 示例二:
[root@localhost sh]# vim xc2.sh

#!/bin/bash

xcf() {
    
    
read -p "请输入:" a
        a=$[$a*2]
        echo $a

}

result=`xcf`
echo $result        ##建议把值放入变量,方便以后调用

mark

补充:

  • 跟在{}下面的为“主体命令”,首先执行的是这个
  • 之后调用函数名,才会执行上面命令序列里面的代码

四、函数传参

  • 顾名思义,传参就是传递参数
  • 之前我们学习过位置变量,这里不再重复
  • 示例如下:
[root@localhost sh]# vim xc3.sh

#!/bin/bash

sum() {
    
    

xcf=$[$1 + $2]
echo $xcf

}

read -p "输入第一个参数:" first
read -p "输入第二个参数:" second

sum $first $second        ##调用函数,传第一个以及第二个输入的值,传参给$1以及$2

mark

  • 另一种更方便的方式
  • 而且我感觉这种方式更容易理解
    mark
    mark
  • 前一种方式显示更加人性化,但对于我们这些专业人员而言,往往后一种用的更多,比较快速高效
  • 更加高端的应用场景往往不需要这么多“人性化”,都是自动化的

五、函数变量的作用范围

  • 函数在Shell脚本中仅在当前Shell环境中有效
  • Shell脚本中变量默认全局有效
  • 将变量限定在函数内部使用local命令
  • 示例如下:
[root@localhost opt]# vim xc1.sh

#!/bin/bash

zxc() {
    
    

a=5
b=6
echo "c等于$c"
}

a=8
c=9
zxc
echo "a等于$a"
echo "b等于$b"
  • 可以看到,在主体命令中定义的c,在函数内也是有效的
  • 为什么先输出c呢,注意主体命令中的顺序,先定义a和c,然后是函数zxc,而在函数zxc里又重新对a和b进行了定义,覆盖了原有定义,再才是zxc下面的输出a和b,所以顺序如下:
[root@localhost opt]# ./xc1.sh 
c等于9
a等于5
b等于6
  • 若我们想把函数定义的值都禁锢在函数当中,需要使用local
  • 仅在函数中生效的变量又称局部变量,下面进行示例:
[root@localhost opt]# vim xc2.sh

#!/bin/bash

zxc() {
    
    
local i
i=8
echo "inside $i"

}

i=9
zxc
echo "outside $i"
  • inside和outside是用来区分的,方便辨识和理解
  • 首先定义i=9,然后输出函数zxc,在函数里定义了i=8,然后输出inside 8,因为这里加了local,所以函数里面定义的8只在函数里有效,而函数外的$i还是之前最开始定义的9
[root@localhost opt]# chmod +x xc2.sh 
[root@localhost opt]# ./xc2.sh 
inside 8
outside 9
  • 再来进一步加深下印象
    mark
    mark
  • 局部变量和全局变量是两种不同的变量,可以赋予不同的值,且是可以同名的,以下看示例讲解
    mark
    mark

六、递归

  • 函数直接或间接调用自己本身的函数
  • 话不多说,开始展示

1.阶乘

科普

  • 阶乘是基斯顿·卡曼于1808 年发明的运算符号,是数学术语
  • 一个正整数的阶乘(factorial)是所有小于及等于该数的正整数的积,并且有0的阶乘为1
自然数n的阶乘写作n!  
n!=1×2×3×…×n

阶乘亦可以递归方式定义:  
0!=1,n!=(n-1)!×n  
n!=n(n-1)(n-2)…1

演示

mark
mark

2.递归目录

  • 递归的本质:
    • 从什么时候开始
    • 到什么时候结束
    • 每次递归需要干什么
  • 示例如下:
  • 遍历/var/log下的所有文件
[root@localhost opt]# vim xc4.sh

#!/bin/bash

function list_files {
    
    
##定义递归遍历目录的函数
for f in `ls $1`
##循环遍历目录
do
        if [ -d "$1/$f" ]
        then
        ##判断若为目录则按格式输出目录名称并继续调用函数遍历这个目录
                echo "$2$f"
                list_files "$1/$f" "$2"
        else
        ##判断若为文件则直接按照格式输出文件名称
                echo "$2$f"
        fi
done
}

list_files "/var/log" ""
##调用函数,第一个参数为要进行遍历的目录,第二个参数为格式设定,区分目录层级
  • 测试一把
[root@localhost opt]# ./xc4.sh 
anaconda
anaconda.log
ifcfg.log
journal.log
ks-script-1_XapN.log
ks-script-b5hX60.log
packaging.log
program.log
storage.log
syslog
X.log
audit
audit.log
boot.log
boot.log-20201222
.
..
...
....
.....略

七、创建函数库

函数库只包含函数的定义,脚本中既包含函数的定义也包括可执行的代码

  • 定义一个函数库,该函数库实现了以下几个函数
    • 加法函数
    • 减法函数
    • 乘法函数
    • 除法函数
  • 首先定义基础函数库文件
[root@localhost opt]# vim xc5.sh

#!/bin/bash

jiafa () {
    
    
        result=$[$1 + $2]
        echo $result
}

jianfa () {
    
    
        result=$[$1 - $2]
        echo $result
}

chengfa () {
    
    
        result=$[$1 * $2]
        echo $result
}

chufa () {
    
    
if [ $2 -ne 0 ]
        then
                result=$[$1 / $2]
                echo $result
        else
                echo "$2不能等于0!"
fi
}

  • 定义脚本
[root@localhost opt]# vim xc6.sh

#!/bin/bash

. /opt/xc5.sh        ##引入函数库文件

read -p "输入第一个参数值:" first
read -p "输入第二个参数值:" second

result1=`jiafa $first $second`
result2=`jianfa $first $second`
result3=$(chengfa $first $second)
result4=$(chufa $first $second)

echo $result1
echo $result2
echo $result3
echo $result4
  • 测试一把
    mark

猜你喜欢

转载自blog.csdn.net/weixin_51486343/article/details/111568761