前言:
在Linux 系统管理的过程中,能够熟练使用shell中的重要工具bash进行编程,在如今运维自动化的趋势中尤为重要。
与其他编程语言相比bash没有属于自己的函数库,bash在实现功能时都是通过PATH路径调用外部的命令,$PATH 路径包含的命令决定了bash可以实现的功能。$PATH路径就是bash的函数库。也就是说bash脚本的编程更像是命令的堆砌,这使得bash编程相对来说更简洁易懂,尤其擅长处理系统管理方面的工作。我们只要熟悉了bash基本特性,就能够写出能够执行复杂任务的高效脚本,在运维工作中,很多问题将会变得事半功倍。本文主要介绍shell编程控制语句并附加相关示例来加深理解与记忆。相关内容如下,请读者自行参阅:
1、if语句
2、case语句
3、for循环及其特殊用法
4、while循环及其特殊用法
5、until循环
6、循环控制语句continue
7、循环控制语句break
8、循环控制shift命令
9、select循环与菜单
10、信号捕捉trap
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 "pelease input your score: " score if [ "$score" -ge 80 -a "$score" -le 100 ];then echo "优秀" elif [ "$score" -lt 80 -a "$score" -ge 60 ];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 "please input yes or no: " yn ans=$(echo $yn|tr "[[:upper:]]" "[[:lower:]]") case $ans in y|yes) echo yes;; n|no) echo no ;; *) echo yes/no? esac
3、for循环
语法:for 变量名 in 列表 ; do
循环体
done
执行机制:依次将列表中的元素赋值给“变量名” ; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环结束。
列表生成方式:
(1) 直接给出列表
(2) 整数列表:
(a) {start..end}
(b) $(seq [start [step]] end)
(3) 返回列表的命令
$(COMMAND)
(4) 使用glob, 如: *.sh
(5) 变量引用;
$@, $*
示例:用for循环打印九九乘法表。
#!/bin/bash for i in {1..9};do for j in $(seq $i);do echo -ne "$j*$i=$[$i*$j]\t" done echo done
for循环的特殊格式:
for ((控制变量初始化 ; 条件判断表达式 ; 控制变量的修正表达式))
do
循环体
done
控制变量初始化:仅在运行到循环代码段时执行一次
控制变量的修正表达式:每轮循环结束会先进行控制变量修正运算,而后再做条件判断
示例:用for循环特殊格式打印九九乘法表。
#!/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
4、while循环
语法:while CONDITION; do
循环体
done
CONDITION:循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为“true”,则执行一次循环;直到条件测试状态为“false”终止循环
因此: CONDTION一般应该有循环控制变量;而此变量的值会在循环体不断地被修正
进入条件: CONDITION为true
退出条件: CONDITION为false
示例:求100以内能被3整除的数之和
#!/bin/bash declare i=1 declare sum=0 while [ "$i" -lt 100 ];do let i++ if [ $[i%3] -ne 0 ];then continue fi let sum+=$i done echo sum is $sum
while的特殊用法(遍历文件的每一行):
while read line; do
循环体
done < /PATH/FROM/SOMEFILE
依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line(也可以是其他自定义变量名)。
示例:找出ID号为偶数的所有用户,显示其用户名及ID号。
while read line ;do if [ $[`echo $line|cut -d: -f3`%2] -eq 0 ];then echo -ne "username:`echo $line|cut -d: -f1` \t" echo "uid:`echo $line|cut -d: -f3`" fi done或者将内容通过管道传给while循环:
示例:若磁盘利用率大于80%,则通知用户。
#!/bin/bash df|grep "/dev/sd"|while read disk ;do diskused=$(echo $disk|sed -r 's/.* ([0-9]+)%.*/\1/') diskname=$(echo $disk|cut -d" " -f1) [ $diskused -ge 80 ] && echo "$diskname will bi full: $diskused%" done
5、untill循环
语法until CONDITION; do
循环体done
进入条件: CONDITION 为false
退出条件: CONDITION 为true示例:用until打印九九乘法表
#!/bin/bash j=1 i=1 until [ "$j" -gt 9 ];do until [ "$i" -gt "$j" ];do echo -ne "$i*$j=$[i*j]\t" let i++ done echo let i=1 let j++ done6、循环控制语句continue
continue用于循环体中
continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层
while CONDTIITON1; do
CMD1
...
if CONDITION2; then
continue
fi
CMDn
...
done示例:打印除了5以外1-10之间的数字。
#!/bin/bash for i in {1..10};do if [ $i -eq 5 ];then continue else echo $i fi done7、循环控制语句break
语法:
break [N]:, 最内层为第1层 while CONDTIITON1; do CMD1 ... if CONDITION2; then break fi CMDn ... donebreak与continue的区别在于,continue是在循环中跳过满足条件的单次循环,而break则默认跳出自己处于的循环体。continue示例中如果换为break则只打印1-4,由于5触发break跳出循环之后的不在打印。
8、循环控制shift命令
shift [n]
n用于将参量列表 list 左移指定次数,缺省为左移一次。
参量列表 list 一旦被移动,最左端的那个参数就从列表中删除。 while 循环遍历位置参量列表时,常用到 shift。
示例#!/bin/bash while (($#>0));do echo $* shift done 显示结果: [root@R1 app]# ./haha.sh a b c d e a b c d e b c d e c d e d e e9、select循环与菜单
select variable in list do 循环体命令 done
select 循环主要用于创建菜单,按数字顺序排列的菜单项将显示在标准错误上,并显示PS3提示符,等待用户输入。
用户输入菜单列表中的某个数字,执行相应的命令。
用户输入被保存在内置变量REPLY中
select 是个无限循环,因此要记住用 break 命令退出循环,或用 exit 命令终止脚本。也可以按 ctrl+c退出循环。
select 经常和 case 联合使用
与 for 循环类似,可以省略 in list, 此时使用位置参量
示例:
#!/bin/bash PS3="please choose your menu: " select menu in huimian lamian hulatang yrt;do case $REPLY in 1) echo "the price is \$10" ;; 2) echo "the price is \$15" ;; 3) echo "the price is \$5" ;; 4) echo "the price is \$20" ;; *) echo "get out!" break esac done
10、信号捕捉trap
trap '触发指令' 信号
自定义进程收到系统发出的指定信号后,将执行触发指令,而不会执行原操作
trap '' 信号
忽略信号的操作
trap '-' 信号
恢复原信号的操作
trap -p
列出自定义信号操作
示例:
#!/bin/bash trap 'echo “signal:SIGINT"' int trap -p for((i=0;i<=10;i++)) do sleep 1 echo $i done trap '' int trap -p for((i=11;i<=20;i++)) do sleep 1 echo $i done trap '-' int trap -p for((i=21;i<=30;i++)) do sleep 1 echo $i done