shell函数
一,函数的定义(两种方式)
1.
function 函数名 {
command
} //这是一种规范写法
2.
函数名(){ //最常用因为最简洁
command
}
函数定义完之后并不会自动执行,需要调用才行,好处在于可以写一段功能代码作为函数,有需要就直接调用
定义的时候哪怕出现语法错误也没关系,不调用就不会报错
当然我们写函数最终目的还是为了调用,为了实现某个功能块
二,函数返回值:
return表示退出函数并返回一个退出值,脚本中可以用$?变量显示该值
使用原则:
1、函数一结束就取返回值,因为$?变量只返回执行的最后一条命令的退出状态码
2、退出状态码必须是0~255,超出时值将为取余256
例
444 % 256
1)函数的调用
直接在脚本里定义函数的代码块后写函数名即可完成调用
#!/bin/bash
function fun1 { //定义了一个函数叫做fun1
echo "this is a function!" //函数体的功能是打印"this is a function!
}
fun1 //直接写函数名就会运行函数体内的代码
[root@server myscripts]# . fun.sh
this is a function!
注意①:函数名必须是唯一,如果先定义了一个,再用同样的名称定义,第二个会覆盖第一个的功能,出现了你不想要的结果,所以这里一定要注意不要重名!
#!/bin/bash
f1 (){
echo hello
}
f1 (){
echo world
}
f1
[root@localhost ~]# . f1.sh
world
注意②:调用函数之前必须先进行定义!
#!/bin/bash
f1 (){
echo hello
}
f3 (){
echo "$(f1) $(f2)"
}
f2 (){
echo world
}
f3
注意③:不一定要在脚本开头就定义函数,只要调用之前定义就可以
如果是在其他地方嵌套调用等情况,不可以直接写函数的值,避免无法识别,可以用反撇号引起来,相当于是调用函数的结果
#!/bin/bash
f1 (){
echo hello
}
f2 (){
echo world
}
f3 (){
echo "`f1` `f2`" //这里如果直接是f1和f2的话,打印出来的不会是hello world而会是f1 f2
}
f3
三,示例
vim localrepo. sh
#!/bin/bash
function backuprepo {
cd /etc/yum.repo.d
mkdir repo.bak
mv * .repo repo.bak
mount /dev/sr0 /mnt > /dev/null
}
makelocalrepo() {
echo '[local]
name= local
baseurl=file: / , /mnt
enabled=1
gpgcheck=0' > local.repo
}
uselocal repo (){
yum clean all > /dev/null
yum makecache > /dev/null
yum install -y httpd > /dev/null
}
######main#######
backuprepo
makelocalrepo
uselocalrepo
四,函数传参
1.两个数求和
#!/bin/bash
sum(){
read -p "请输入第一个数:" NUM1
read -p "请输入第二个数:" NUM2
echo "你输入的两个数为: $NUM1 和$NUM2 "
SUM=$(( NUM1+$NUM2))
echo “两个数的和为: $SUM”
}
sum
1.函数作用范围
在 Shell 脚本中函数的执行并不会开启一个新的子 Shell,而是仅在当前定义的 Shell 环境中有效。如果Shell脚本中的变量没有经过特殊设定,默认在整个脚本中都是有效的。在编写脚本时,有时需要将变量的值限定在函数内部,可以通过内置命令local来实现。函数内部变量的使用,可以避免函数内外同时出现同名变量对脚本结果的影响。
shell脚本中变量默认全局有效
local命令:将变量限定在函数内部使用
[root@localhost ~]# vim myfun.sh
myfun ()
{
local i
i=8
echo $i
}
i=9
myfun
echo $i
上述脚本中myfun 函数内部使用了local命令设置变量i,其作用是将变量i限定在函数内部。myfun 函数外部同样定义了变量i,内部变量i和全局变量i互不影响。脚本执行时先调用了函数myfun,函数内部变量i为8,所以输出结果是8。调用完函数之后,给变量i赋值为9,再打印外部变量i,所以又输出 9
myfunc() {
a=8
echo $a
}
myfunc
echo $a
#!/bin/bash
myfunc() {
a=8
echo $a
myfunc
a=9
echo $a I
2.传参
函数的传参
1)函数的参数
计算位置变量$1和$2的和
#!/bin/bash
add (){
let sum=$1+$2 //这里的位置变量是函数的位置变量,所以要写在调用函数的后面,如果是调用脚本时使用则 不能成功
echo $sum
}
add 4 5
2)脚本的参数
#!/bin/bash
add (){
let sum=$1+$2
echo $sum
}
add $1 $2 //这里相当于调用了脚本的参数了,然后把脚本的位置变量传递给函数进行计算
通过脚本传递参数给函数中的位置参数$1
#!/bin/bash
fun1(){
rm -rf $1
}
fun1 $1
调用函数时直接传递参数
#!/bin/bash fun1(){
rm -rf $1
}
fun1 /root/a.txt
函数中多参数传递和使用方法
3.函数的递归
函数自己调用自己的本身
列出目录内文件列表,目录用蓝色表示,文件显示层级关系
#!/bin/bash
list(){
for i in $1/*
do
if [ -d $i ];then
echo -e "\e[34m$i\e[0m"
list $i " $2"
else
echo "$2$i"
fi
done
}
list $1 $2
计算5的阶乘
#阶乘 5! 5*4*3*2*1
#!/bin/bash
fa () {
if [ $1 -eq 1 ]
then
echo 1
else
local tp=$[ $1 - 1]
local res=$(fa $tp)
echo $[ $1 * $res ]
fi
}
五,本地变量与全局变量
在脚本里定义的变量或者在函数体没有声明为本地变量的都为全局变量,意思是在当前shell环境都识别
如果需要这个变量只在函数中使用则可以在函数中用local关键字声明,这样即使函数体外有个重名的变量也没关系,不影响在函数体的使用
如果是用source执行脚本的话就能看出差别
[root@localhost ~]# a=nihao
#!/bin/bash //写一个函数,首先打印a的值,然后改变a的值再打印
f1 (){
echo $a
a=world
echo $a
}
f1
[root@localhost ~]# source f1.sh //用source执行脚本,会发现a的值改变了
nihao
world
[root@localhost ~]# echo $a
world
在函数体中用local定义变量a的值
#!/bin/bash
f1 (){
echo $a
local a=world
echo $a
}
f1
[root@localhost ~]# source f1.sh //执行脚本发现a的值并没有改变
world
world
#!/bin/bash
f1 (){
echo $a
local a=world
echo $a
}
f1
echo $a