shell脚本编程详解3:
在前两节我们分别介绍了shell脚本编程的基础和循环,判断的知识,今天我们分享脚本的另一种用法,函数.
概论:
函数function是由若干条shell命令组成的语句块,实现代码重用和模块化编程。
它与shell程序形式上是相似的,不同的是它不是一个单独的进程,不能独立运行,而是shell程序的一部分。
函数和shell程序比较相似,区别在于:
Shell程序在子Shell中运行
而Shell函数在当前Shell中运行。因此在当前Shell中,函数可以对shell中变量进行修改
使用:
语法:
函数由两部分组成:函数名和函数体。
语法一:
function f_name {
…函数体…
}
语法二:
f_name() {
…函数体…
}
定义:
可在交互式环境下定义函数
可将函数放在脚本文件中作为它的一部分
可放在只包含函数的单独文件中
调用:函数只有被调用才会执行;
调用:给定函数名
函数名出现的地方,会被自动替换为函数代码
函数的生命周期:被调用时创建,返回时终止
返回值:
函数有两种返回值:
函数的执行结果返回值:
(1) 使用echo或printf命令进行输出
(2) 函数体中调用命令的输出结果
函数的退出状态码:
(1) 默认取决于函数中执行的最后一条命令的退出状态码
(2) 自定义退出状态码, 其格式为:
return 从函数中返回,用最后状态命令决定返回值
return 1-255 有错误返回
交互式使用函数:
示例:
$dir() {
> ls -l
> }
定义该函数后,若在$后面键入dir,其显示结果同ls -l的
作用相同。
$dir
该dir函数将一直保留到用户从系统退出,或执行了如下
所示的unset命令:
$ unset dir
脚本使用函数:
函数在使用前必须定义,因此应将函数定义放在脚本开始部分,直至
shell首次发现它后才能使用
调用函数仅使用其函数名即可。
示例:
$cat func1
#!/bin/bash
# func1
hello()
{
echo "Hello there today's date is `date +%F`"
}
echo "now going to the function hello"
hello
echo "back from the function"
函数文件:
可以将经常使用的函数存入函数文件,然后将函数文件载入shell。
文件名可任意选取,但最好与相关任务有某种联系。例如: functions.main
一旦函数文件载入shell,就可以在命令行或脚本中调用函数。可以使用set命令查看所有定义的函数,其输出列表包括已经载入shell的所有函数。
若要改动函数,首先用unset命令从shell中删除函数。改动完毕后,再重新载入此文件
创建函数文件:
$cat functions.main
#!/bin/bash
#functions.main
findit()
{
if [ $# -lt 1 ] ; then
echo "Usage:findit file"
return 1
fi
find / -name $1 –print
}
载入函数:
函数文件已创建好后,要将它载入shell
定位函数文件并载入shell的格式:
. filename 或 source filename
注意:此即<点> <空格> <文件名>
这里的文件名要带正确路径
示例:上例中的函数,可使用如下命令:
$ . functions.main
检查载入函数:
使用set命令检查函数是否已载入。 set命令将在shell中显示所有的载入函数。
示例:
$set
findit=( )
{
if [ $# -lt 1 ]; then
echo "usage :findit file";
return 1
fi
find / -name $1 -print
}
…
执行shell函数:
要执行函数,简单地键入函数名即可:
示例:
$findit groups /usr/bin/groups /usr/local/backups/groups.bak
删除shell函数:
现在对函数做一些改动。首先删除函数,使其对shell不可用。使用unset命令完成此功能.
命令格式为:
unset function_name
实例:
$unset findit
再键入set命令,函数将不再显示
函数参数:
函数可以接受参数:
传递参数给函数:调用函数时,在函数名后面以空白分隔
给定参数列表即可;例如“ testfunc arg1 arg2 …”
在函数体中当中,可使用$1, $2, …调用这些参数;还
可以使用$@, $*, $#等特殊变量
函数变量:
变量作用域:
环境变量:当前shell和子shell有效
本地变量:只在当前shell进程有效,为执行脚本会启动
专用子shell进程;因此,本地变量的作用范围是当前shell脚本
程序文件,包括脚本中的函数。
局部变量:函数的生命周期;函数结束时变量被自动销毁
注意:如果函数中有局部变量,如果其名称同本地变量, 使用局部变量。
在函数中定义局部变量的方法
local NAME=VALUE
函数递归:
函数递归:
函数直接或间接调用自身
注意递归层数
递归实例:
阶乘是基斯顿·卡曼于 1808 年发明的运算符号,是数学术语
一个正整数的阶乘( factorial)是所有小于及等于该数的正整
数的积,并且有0的阶乘为1。自然数n的阶乘写作n!。
n!=1×2×3×…×n。
阶乘亦可以递归方式定义: 0!=1, n!=(n-1)!×n。
n!=n(n-1)(n-2)…1
n(n-1)! = n(n-1)(n-2)!
栗子:
fact.sh
#!/bin/bash
#
fact() {
if [ $1 -eq 0 -o $1 -eq 1 ]; then
echo 1
else
echo $[$1*$(fact $[$1-1])]
fi
}
fact 5
实战:
1.加减乘除函数,脚本调用
#!/bin/bash
jia(){
sum=0
sum=$[$1+$2]
echo $sum
}
jian(){
sum=0
sum=$[$1-$2]
echo $sum
}
cheng(){
sum=0
sum=$[$1*$2]
echo $sum
}
chu(){
sum=0
sum=$[$1/$2]
echo $sum
}
#!/bin/bash
#****************************************************
#Description: 测试加减乘除函数脚本
#****************************************************
. sum.sh
echo ' 6=$1,3=$2'
jia 6 3
jian 6 3
cheng 6 3
chu 6 3
2.管理服务脚本
1、编写服务脚本/root/bin/testsrv.sh,完成如下要求
(1) 脚本可接受参数:start, stop, restart, status
(2) 如果参数非此四者之一,提示使用格式后报错退出
(3) 如是start:则创建/var/lock/subsys/SCRIPT_NAME, 并显示“启动成功”
考虑:如果事先已经启动过一次,该如何处理?
(4) 如是stop:则删除/var/lock/subsys/SCRIPT_NAME, 并显示“停止完成”
考虑:如果事先已然停止过了,该如何处理?
(5) 如是restart,则先stop, 再start
考虑:如果本来没有start,如何处理?
(6) 如是status, 则如果/var/lock/subsys/SCRIPT_NAME文件存在,则显示“SCRIPT_NAMEis running…”
如果/var/lock/subsys/SCRIPT_NAME文件不存在,则显示“SCRIPT_NAME is stopped…”
其中:SCRIPT_NAME为当前脚本名
#!/bin/bash
#
#chkconfig:35 66 33
#description:this is myservice
lock=/var/lock/subsys/myservice
start(){
if [ ! -e $lock ];then
touch $lock
echo -e "start myservice [ \033[32mOK\033[0m ]"
else
echo -e "start myservice [ \033[31mFAILED\033[0m ]"
fi
}
stop(){
if [ -e $lock ];then
rm -rf $lock
echo -e "stop myservice [ \033[32mOK\033[0m ]"
else
echo -e "stop myservice [ \033[31mFAILED\033[0m ]"
fi
}
status(){
if [ -e $lock ];then
echo "myservice is running"
else
echo "myservice alread stopped"
fi
}
case $i in
start)
start;;
stop)
stop;;
restart)
stop
start;;
status)
status;;
*)
echo "Usage: `basename $0` start|stop|restart|status";;
esac
3
编写脚本/root/bin/copycmd.sh
(1) 提示用户输入一个可执行命令名称;
(2) 获取此命令所依赖到的所有库文件列表
(3) 复制命令至某目标目录(例如/mnt/sysroot)下的对应路径下;
如:/bin/bash ==> /mnt/sysroot/bin/bash
/usr/bin/passwd==> /mnt/sysroot/usr/bin/passwd
(4) 复制此命令依赖到的所有库文件至目标目录下的对应路径下:
如:/lib64/ld-linux-x86-64.so.2 ==> /mnt/sysroot/lib64/ld-linux-x86-64.so.2
(5)每次复制完成一个命令后,不要退出,而是提示用户键入新的要复制的命令,并重复完成上述功能;直到用户输入quit退出
#!/bin/bash
copy(){
if [ ! -e $dir$cmddir_path ];then
mkdir -p $dir$cmddir_path &>/dev/null
cp -r $cmddir $dir$cmddir_path &>/dev/null
echo " $cmd file no"
elif [ -e $dir$cmddir ];then
echo " $cmd file is exit!!!"
elif [ -e $dir$cmddir_path ];then
cp -r $cmddir $dir$cmddir_path
echo "$cmd file copy over!!"
else
echo "A"
fi
}
copylib(){
for i in $libdir
do
lib=`echo $i|sed 's@.*=>@@'| tr -s "\t" " "|cut -d' ' -f2`
lib_path=`echo $i|sed 's@[^/]\+$@@'`
if [ ! -e $lib ];then
echo "$lib is no exit"
elif [ ! -e $dir$lib_path ];then
mkdir -p $dir$lib_path &>/dev/null
cp $lib $dir$lib_path &>/dev/null
echo "$cmd lib dir is no ,mkdir filedir,and copy libfile"
elif [ -e $dir$lib_path ];then
cp $lib $dir$lib_path &>/dev/null
echo "$cmd lib file copy over!!!"
fi
done
}
while true
do
read -p "Please Enter A Command(quit:exit the script) :" cmd
dir=/mnt/sysroot
cmddir=`which --skip-alias $cmd`
cmddir_path=`echo $cmddir|sed 's@[^/]\+$@@'`
libdir=`ldd $cmddir`
if [ -z $cmd ];then
echo " at least enter a cmd!!!"
continue
elif [ "$cmd" == "quit" ];then
echo " soon exit a script "
exit 2
elif [! $cmddir &>/dev/null ];then
echo "this commend is wrong,again try..."
continue
else
echo "copt $cmd started at once"
#copy
copylib
fi
done
原创文章,作者:wencx,如若转载,请注明出处:http://www.178linux.com/37729


评论列表(1条)
shell脚本函数不仅能实现代码重用,还能提高运行效率。linux中很多程序,都是使用函数来进行使用和调用的,所谓的组合小程序,完成复杂任务就是这样的思想。希望下来能多加练习,培养出模块化的思想哦。