前言:
在Linux 系统管理的过程中,能够熟练使用shell中的重要工具bash进行编程,在如今运维自动化的趋势中尤为重要。
与其他编程语言相比bash没有属于自己的函数库,bash在实现功能时都是通过PATH路径调用外部的命令,$PATH 路径包含的命令决定了bash可以实现的功能。$PATH路径就是bash的函数库。也就是说bash脚本的编程更像是命令的堆砌,这使得bash编程相对来说更简洁易懂,尤其擅长处理系统管理方面的工作。我们只要熟悉了bash基本特性,就能够写出能够执行复杂任务的高效脚本,在运维工作中,很多问题将会变得事半功倍。本文主要介绍shell编程控制语句并附加相关示例来加深理解与记忆。相关内容如下,请读者自行参阅:
1、if语句
1、条件选择:if语句
①单分支
if判断条件;then 条件为真的分支代码 fi
②双分支
if判断条件;then 条件为真的分支代码 else 条件为假的分支代码 fi
③多分支
if判断条件1;then 条件为真的分支代码 elif判断条件2;then 条件为真的分支代码 elif判断条件3;then 条件为真的分支代码 else 以上条件都为假的分支代码 fi
示例:请用户输入成绩:80-100为优秀,60-79为通过,其他淘汰。
#!/bin/bash read-p"peleaseinputyourscore:"score if["$score"-ge80-a"$score"-le100];then echo"优秀" elif["$score"-lt80-a"$score"-ge60];then echo"通过" else echo"淘汰" fi
2、条件判断: case语句
语法:case变量引用in PAT1) 分支1;; PAT2) 分支2;; ... *) 默认分支;; esac
case支持glob风格的通配符:
*: 任意长度任意字符
?: 任意单个字符
[]:指定范围内的任意单个字符
a|b: a或b
。。。
示例:编写脚本,提示用户输入yes或no不区分大小写,并判断用户输入的是yes还是no或者其他信息。
#!/bin/bash read-p"pleaseinputyesorno:"yn ans=$(echo$yn|tr"[[:upper:]]""[[:lower:]]") case$ansin y|yes) echoyes;; n|no) echono;; *) echoyes/no? esac
语法:for 变量名 in 列表; do
循环体
done
执行机制:依次将列表中的元素赋值给“变量名” ; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环结束。
列表生成方式:
(1) 直接给出列表
(2) 整数列表:
(a) {start..end}
(b) $(seq [start [step]] end)
(3) 返回列表的命令
$(COMMAND)
(4) 使用glob, 如: *.sh
(5) 变量引用;
$@,$*
示例:用for循环打印九九乘法表。
#!/bin/bash foriin{1..9};do forjin$(seq$i);do echo-ne"$j*$i=$[$i*$j]\t" done echo done
for循环的特殊格式:
for ((控制变量初始化;条件判断表达式; 控制变量的修正表达式))
do
循环体
done
控制变量初始化:仅在运行到循环代码段时执行一次
@H_403_170@控制变量的修正表达式:每轮循环结束会先进行控制变量修正运算,而后再做条件判断
示例:用for循环特殊格式打印九九乘法表。
@H_403_170@
#!/bin/bash for((i=1;i<=9;i++));do for((j=1;j<=i;j++));do echo-ne"$j*$i=$[$j*$i]\t" done echo done
语法:while CONDITION; do
循环体
done
CONDITION:循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为“true”,则执行一次循环;直到条件测试状态为“false”终止循环
因此: CONDTION一般应该有循环控制变量;而此变量的值会在循环体不断地被修正
进入条件: CONDITION为true
退出条件: CONDITION为false
示例:求100以内能被3整除的数之和
#!/bin/bash declarei=1 declaresum=0 while["$i"-lt100];do leti++ if[$[i%3]-ne0];then continue fi letsum+=$i done echosumis$sum
while read line; do
循环体
done < /PATH/FROM/SOMEFILE
依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line(也可以是其他自定义变量名)。
whilereadline;do if[$[`echo$line|cut-d:-f3`%2]-eq0];then echo-ne"username:`echo$line|cut-d:-f1`\t" echo"uid:`echo$line|cut-d:-f3`" fi done</etc/passwd
或者将内容通过管道传给while循环:
#!/bin/bash df|grep"/dev/sd"|whilereaddisk;do diskused=$(echo$disk|sed-r's/.*([0-9]+)%.*/\1/') diskname=$(echo$disk|cut-d""-f1) [$diskused-ge80]&&echo"$disknamewillbifull:$diskused%" done
语法until CONDITION; do
循环体
done
进入条件: CONDITION 为false
退出条件: CONDITION 为true
示例:用until打印九九乘法表
#!/bin/bash j=1 i=1 until["$j"-gt9];do until["$i"-gt"$j"];do echo-ne"$i*$j=$[i*j]\t" leti++ done echo leti=1 letj++ done
continue用于循环体中
continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层
while CONDTIITON1; do
CMD1
...
if CONDITION2; then
continue
fi
CMDn
...
done
示例:打印除了5以外1-10之间的数字。
#!/bin/bash foriin{1..10};do if[$i-eq5];then continue else echo$i fi done
语法:
break[N]:,最内层为第1层 whileCONDTIITON1;do CMD1 ... ifCONDITION2;then break fi CMDn ... done
break与continue的区别在于,continue是在循环中跳过满足条件的单次循环,而break则默认跳出自己处于的循环体。continue示例中如果换为break则只打印1-4,由于5触发break跳出循环之后的不在打印。
shift [n]
n用于将参量列表 list 左移指定次数,缺省为左移一次。
参量列表 list 一旦被移动,最左端的那个参数就从列表中删除。 while 循环遍历位置参量列表时,常用到 shift。
示例
#!/bin/bash while(($#>0));do echo$* shift done 显示结果: [root@R1app]#./haha.shabcde abcde bcde cde de e
9、select循环与菜单
selectvariableinlist do 循环体命令 done
select 循环主要用于创建菜单,按数字顺序排列的菜单项将显示在标准错误上,并显示PS3提示符,等待用户输入。
@H_403_170@- @H_403_170@
用户输入被保存在内置变量REPLY中
@H_403_170@select 是个无限循环,因此要记住用 break 命令退出循环,或用 exit 命令终止脚本。也可以按 ctrl+c退出循环。
@H_403_170@select 经常和 case 联合使用
@H_403_170@与 for 循环类似,可以省略 in list, 此时使用位置参量
示例:
@H_403_170@
#!/bin/bash PS3="pleasechooseyourmenu:" selectmenuinhuimianlamianhulatangyrt;do case$REPLYin 1) echo"thepriceis\$10";; 2) echo"thepriceis\$15";; 3) echo"thepriceis\$5";; 4) echo"thepriceis\$20";; *) echo"getout!" break esac done
trap '触发指令' 信号
自定义进程收到系统发出的指定信号后,将执行触发指令,而不会执行原操作
trap '' 信号
忽略信号的操作
trap '-' 信号
恢复原信号的操作
trap -p
列出自定义信号操作
示例:
#!/bin/bash trap'echo“signal:SIGINT"'int trap-p for((i=0;i<=10;i++)) do sleep1 echo$i done trap''int trap-p for((i=11;i<=20;i++)) do sleep1 echo$i done trap'-'int trap-p for((i=21;i<=30;i++)) do sleep1 echo$i done