目录
1、Shell函数作用
将命令序列按格式写在一起可方便重复使用命令序列避免代码重复
使用函数可以将大的工程分割为若干小的功能模块,代码的可读性更强
2、Shell函数定义
function 函数名(){
命令序列
[return x] #使用return或exit可以显示的结束函数
}
或者
函数名 (){
命令序列
[return x] #使用return或exit可以显示的结束函数
}
注意事项
- 直接写 函数中调用函数 直接写函数名
- 同名函数 后一个生效
- 调用函数一定要先定义
- 只要先定义了调用的 其他函数定义顺序无关
3、调用函数的方法
函数名 [参数1][参数2]
4、Shell函数返回值
return表示退出函数并返回一个退出值,脚本中可以用$?变量显示该值
使用原则:
1、函数一结束就取返回值,因为$?变量只返回执行的最后一条命令的退出状态码
2、退出状态码必须是0~255,超出时值将为除以256取余
解决数值超出255的办法
[root@localhost hanshu]# vim 3.sh
#!/bin/bash
test1 (){
read -p "请输入一个整数:" num
echo $[num * 2]
}
###########main############
result=$(test1)
echo $result
[root@localhost hanshu]# bash 3.sh
请输入一个整数:300
600
5、函数传参
在Shell中,调用函数时可以向其传递参数。在函数体内部,通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数…即使用位置参数来实现参数传递。
示例1:
[root@localhost hanshu]# vim 1.sh
#!/bin/bash
#文件上方先定义函数,然后在主代码部分调用函数
test1 () {
sum=$[ $1 + $2 ]
echo $sum
}
##############main#################
test1 10 20
[root@localhost hanshu]# bash 1.sh
30
示例2:
[root@localhost hanshu]# vim 1.sh
#!/bin/bash
test1 () {
#函数中的$1 $2代表调用函数时 函数后面跟的位置参数
echo $1
echo $2
}
##############main#################
#主体代码中的$1 $2 代表执行脚本时,脚本后面跟的位置参数
test1 $2 $1
[root@localhost hanshu]# bash 1.sh 3 4
4
3
6、函数变量的作用范围
- 函数在Shell脚本中仅在当前Shell环境中有效
- Shell脚本中变量默认全局有效
- 将变最限定在函数内部使用local命令
示例1:Shell脚本中便量全局用效
[root@localhost hanshu]# vim 4.sh
#!/bin/bash
fun(){
i=10
}
#########main#########
i=20
fun
echo $i
[root@localhost hanshu]# bash 4.sh
10
示例2、使用local命令
[root@localhost hanshu]# vim 4.sh
#!/bin/bash
fun(){
local i=10
echo $i
}
#########main#########
i=20
fun
echo $i
[root@localhost hanshu]# bash 4.sh
10
20
7、函数的递归
函数调用自己本身
示例1:阶乘
[root@localhost ~]# vim jiecheng.sh
#!/bin/bash
#使用递归计算阶乘
fact() {
if [ $1 -eq 1 ] ;then
echo 1
else
local temp=$[$1 - 1]
local result=$(fact $temp )
echo $[$1 * $result]
fi
}
##############main###############
read -p"请输入一个正整数:" num
last=`fact $num`
echo "$num的阶乘的结果为:$last"
[root@localhost ~]# bash jiecheng.sh
请输入一个正整数:5
5的阶乘的结果为:120
示例2 查看/root/bin下的所有子目录和文件
方法1:
[root@localhost ~]# vim ml.sh
#!/bin/bash
#递归/ root/bin目录,显示它的所有子目录和文件
list() {
for fd in $1/*
do
if [ -d $fd ];then
echo "$fd是目录"
list "$fd"
else
echo "$fd 是文件"
fi
done
}
########main#####
list "/root/bin"
[root@localhost ~]# bash ml.sh
/root/bin/1.txt 是文件
/root/bin/aa是目录
/root/bin/aa/bb是目录
/root/bin/aa/bb/* 是文件
方法2:
[root@localhost ~]# vim ml.sh
#!/bin/bash
#递归/ root/bin目录,显示它的所有子目录和文件
list() {
#逐个检查$1参数指定目录下的所有文件和子目录
for fd in $(ls $1)
do
#判断如果是目录就输出,并且通过把这个目录下的所有文件或子目录再诸葛检查,如果发现又子目录就会按
照这个方式一直检查下去
if [ -d "$1/$fd" ];then
echo "$fd是目录"
#在递归函数调用时第二个位置参数$2前加的两个空格是为了让每一次帝国都在文件名前多加两个空格,使
输出看起来有层级关系
list "$1/$fd" " $2"
else
echo "$2$fd 是文件"
fi
done
}
########main#####
list "/root/bin" ""
[root@localhost ~]# bash ml.sh
1.txt 是文件
aa是目录
bb是目录
cc.txt 是文件
8、函数库
可以事先创建一个函数库文件,在里面定义各种常用的函数,然后可以在别的shell脚本中直接引用这个函数库文件,使得不需要再次定义函数即可直接调用函数
[root@localhost ~]# vim ku.sh
#!/bin/bash
jiafa() {
echo $[$1 + $2]
}
jianfa() {
echo $[$1 - $2]
}
chengfa() {
echo $[$1 * $2]
}
chufa(){
if [ $2 -eq 0 ] ;then
echo"除数不能为0"
else
echo $[$1 / $2]
fi
}
jiecheng(){
if [ $1 -eq 1 ] ; then
echo 1
else
local temp=$[$1 - 1]
local result=$(jiecheng $temp)
echo $[$1 * $result]
fi
}
#不需要定义 直接使用函数库文件
[root@localhost ~]# vim 10.sh
#!/bin/bash
. ku.sh
v1=10
v2=5
res1=$(jiafa $v1 $v2)
res2=$(jianfa $v1 $v2)
res3=$(chengfa $v1 $v2)
res4=$(chufa $v1 $v2)
res5=$(jiecheng $v1)
echo "加法结果为:$res1"
echo "减法结果为:$res2"
echo "乘法结果为:$res3"
echo "除法结果为:$res4"
echo "${v1}阶乘结果为:$res5"
[root@localhost ~]# bash 10.sh
加法结果为:15
减法结果为:5
乘法结果为:50
除法结果为:2
10阶乘结果为:3628800