![高性能Linux服务器运维实战:shell编程、监控告警、性能优化与实战案例](https://wfqqreader-1252317822.image.myqcloud.com/cover/769/33643769/b_33643769.jpg)
2.4 while循环、until循环以及select循环
2.4.1 while循环结构
1.while循环语句
while循环语句的语法结构如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/90_02.jpg?sign=1739663791-lXMuJoc6LQ56NyIUajuPND0Vbdqj8R6i-0-e843d6c96043d1d03e19fd73fab09c20)
while的执行过程为:先执行expr,如果其退出状态为0,就执行循环体。执行到关键字done后,回到循环的顶部,while命令再次检查expr的退出状态。以此类推,循环将一直继续下去,直到expr的退出状态非0为止。
2.while循环语句执行流程
while循环语句的执行流程如图2-6所示。
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/90_03.jpg?sign=1739663791-jwDkzAtzR4mroFsHqlQC7UW20idne1uM-0-4dad9d7c2a0b9ee70a1cbe6523313436)
图2-6 while循环执行流程
从图中可以看出:while循环首先会进行条件测试判断,如果条件为真,那么执行do中的命令,执行完毕,继续进行while循环中的条件测试判断,如果为真,继续执行do中的命令,如果为假,那么结束循环。
while循环语句也称前测试循环语句,它的循环重复执行次数是利用一个条件来控制是否继续重复执行这个语句。
while循环语句之所以命名为前测试循环,是因为它要先判断此循环的条件是否成立,然后才作重复执行的操作。也就是说,while循环语句的执行过程是:先判断expr表达式的退出状态,如果退出状态为0,则执行循环体,执行到关键字done后,回到循环的顶部,进行下一次循环,否则退出循环,执行done后的命令。为了避免死循环,必须保证在循环体中包含循环出口条件,即存在expr表达式的退出状态为非0的情况。
3.while循环例子
下面通过几个实例来熟悉和深入了解while的用法和功能。
(1)最基本的while循环
看下面这个基于while循环的shell脚本while1.sh,内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/91_01.jpg?sign=1739663791-EpXWaSqJpcU00jDbWhU8sAxG1wQGHRQZ-0-1bbadc871fd9a66bb7009c8cb11a20d6)
这个脚本实现的功能是猜数字游戏。$RANDOM是一个系统随机数的环境变量,模100运算用于生成1~100的随机整数,通过$((RANDOM%100))生成一个随机整数,然后和输入的数字进行比较,比较分3种情况,输入数字小于、大于和等于随机整数,如果输入数字等于随机整数,那么执行break退出循环。
执行这个脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/91_02.jpg?sign=1739663791-tlnZgakyy9t26Bip9ET2yEeWAnzDzjrN-0-b4abb36d06328a243d03793268d94750)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/92_01.jpg?sign=1739663791-dpfeKWhkZR6CMj1m9qLsZEbEdbbZ08AP-0-179b44efd7d2f99e627b1aa85586b1cd)
(2)while从文件读取内容赋给指定变量
继续看下面这个例子,while2.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/92_02.jpg?sign=1739663791-yYSayBW3kGcLYZaAGRUAI2qIIZ56rShh-0-6bbe1a1a98f6227d03b61822f2b8fa79)
这个while循环是通过读取文件内容,然后通过指定分隔符,将分割出来的内容赋给read后面给出的7个变量,然后在循环体中进行判断,最后,输出满足条件的内容。
执行这个脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/92_03.jpg?sign=1739663791-JZWb3ItqsQoYkeYfb1HA5pQ1mkAY1njm-0-1ffd71e85b60f6afc92663ba85cf5d61)
(3)while配合read读取文件
最常用的方法是对文件使用cat命令并通过管道将结果直接传送给包含read命令的while命令,每次调用read命令都会读取文件中的“一行”文本。当文件没有可读的行时,read命令将以非零状态退出。while2.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/92_04.jpg?sign=1739663791-eofEpqaWXkXMfsQ0whLKARMQuHDIgSMR-0-aed961d97e04db322af3938e7e8ed877)
此外,还可以用如下方式:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/92_05.jpg?sign=1739663791-TEG6gffnMMjENyarCPjFUAFvMVzBjV3R-0-dacc76bf754da17e06c7a834c34302a2)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/93_01.jpg?sign=1739663791-W5vjNOCONyfC6d5a2uRO1E2bJOdJ7KgY-0-0a26549e6b29e02a5348ceec8b3f98fd)
这个脚本实现的结果跟上面那个脚本完全一样,不同的是读取文件的方式,这个脚本是通过重定向直接读取文件内容的。执行这个脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/93_02.jpg?sign=1739663791-YNuHX9tro31bTNzvhYNY1YBlDKNdCBSx-0-baecc77851c4aed2bf34a31846ed9f19)
(4)while与管道配合使用
看下面这个例子,rename_filename.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/93_03.jpg?sign=1739663791-PFVtGKJ72Sr8QENLKcBK42RdgTtr9hDA-0-142c5e4495395faf5e248e59ae0acacb)
这个脚本的功能是找出当前目录下包含空格的文件名,将空格替换为下划线。它将find命令查找的结果通过管道交给while循环,然后while通过读取find得到的结果进行判断,如果read读到的文件名有空格,那么将空格替换为下划线。执行这个脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/93_04.jpg?sign=1739663791-BjFVpwJIcfS4eCJJBy5fwd5iY05y2icX-0-89db3e9078806b4b926e815fd85a9fae)
从输出结果可看到,带空格的文件名中的空格已经被替换成了下划线。
4.计数器控制的while循环
如果已经准确知道要输入的数据或字符串的数目,可采用计数器控制的while循环结构来处理。while循环的格式如下所示:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/93_05.jpg?sign=1739663791-DrajcaCZznOCZ4fqTuB2b7Qhm6fvwxUW-0-e8a2283837c2e294ed69b421ac4c996f)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/94_01.jpg?sign=1739663791-8ul7tDfa6JlQDYCEPgTXhmpQcaR3U2V2-0-332eb93cd363d06f84ce437bc15344dc)
这个while循环结构,首先会有一个初始值,例如,这里的counter=1,然后开始执行while循环,在while循环体中,会对counter的值进行改变,例如,这里通过let改变counter的值,接着继续执行循环体,执行完成。再次返回while循环的expression进行判断,如果满足条件,继续执行循环体,如果不满足,则退出循环。
来看下面这个例子,while3.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/94_02.jpg?sign=1739663791-KcGVtEGDFb4EDI1JoxhSzgWog4d3qa92-0-a5900accdc35980d1ed66ef6358bbf8a)
执行这个脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/94_03.jpg?sign=1739663791-RF5bT27WK6pm3i195u0B6i2uEE5Tan1V-0-484a9c9c8610adfa2f251055dfbe95dd)
这里注意脚本中循环体里面的echo $count,如果echo $count写在循环体最上面,那么输出结果将是1~9,如果写到循环体最下面,那么输出结果就是2~10。
5.结束标记控制的while循环
在Linux shell编程中很多情况下不知道读入数据的个数,但是可以设置一个特殊的数值来结束while循环,该特殊数值称为结束标记,其通过提示用户输入特殊字符或数字来操作。当用户输入该标记后结束while循环,执行done后的命令。在该情形下,while循环的形式如下所示:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/95_01.jpg?sign=1739663791-IdeuWGW9d74W2UGJhVFD4iBi9LzAqB9I-0-b19316cfe0233ecf387fb53ae9a4c679)
在这个while循环中,首先会读入用户的输入,然后进入while循环的条件判断,如果满足条件,则进入循环体,循环体仍然是读取用户的输入,就这样反复循环,直到输入的内容不满足while循环的条件,则退出循环。执行done后的命令。
下面看一个例子,脚本while4.sh内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/95_02.jpg?sign=1739663791-LSxEXB6lpcKk2DNxQ1DVCLS0tg33rEZl-0-685ba147214ee14a885b0f8c7deb9b1d)
这个脚本是读入用户输入,如果用户输入的内容为非OK,那么循环一直持续下去,否则,当输入内容为OK时,循环结束。执行这个脚本,输出内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/95_03.jpg?sign=1739663791-FK9yHPGyKGw6ryjWWTYdpZsGYIZUZtHx-0-9dcd5f204d4e9b362072bef9f13bbd41)
看这个输出过程,有助于理解这个while循环的执行过程。当用户第1次输入www的时候,才进入while循环,判断条件为真后,进入循环体,执行循环体里面的read命令,进而显示循环体里面读入的用户输入(非循环体外的read读入值),所以第1个输入www,仅出现一次。
6.标志控制的while循环
标志控制的while循环是使用用户输入的标志的值来控制循环的结束,这样避免了用户不知道循环结束标记的麻烦。在该情形下,while循环的形式如下所示:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/95_04.jpg?sign=1739663791-FbZoAELWZ1FP2mhCOjQIq94PRQ2B43JL-0-3480402a614313c3053b50152641dea7)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/96_01.jpg?sign=1739663791-DbbluZ3b01fg6pkosGBXLpuigul6quep-0-84336e4a48871f022f2ac23634861544)
在这个while循环中,首先设置signal为0,然后进入while循环体。while的判断条件是signal不等于1的情况,所以满足条件,进入循环体。在循环体中嵌套了一个if判断,当if后面的条件满足时,就将signal置为1,下次返回while循环进行条件判断时,发现条件为假,这样就退出了while循环,执行done后面的操作。
下面看一个例子,while5.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/96_02.jpg?sign=1739663791-8PcOy4X1R1UsxXaND2g0kMropgIkq4gj-0-b08cf2f6458089b453f9aa480c744e70)
这个shell脚本中,首先将signal置为0,然后进入while循环体。在while循环体中有个if条件判断,这个条件判断是读取输入的参数$@,如果输入的参数是ok,那么就将signal置为1,这样下次进入while判断的时候,就会发现条件为假,进而退出while循环。
执行这个脚本,输出内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/96_03.jpg?sign=1739663791-HhhnOF4O3Vo1e35dsnUexJHI879A3ykr-0-d68878834efbd30795d6d8738dbaa3aa)
第1次执行while5.sh脚本时,输入的参数是www,由于不满足if语句的条件,所以不会执行if里面的操作,因此signal始终为0,这样就进入了while的无限循环中。第2次执行while5.sh脚本时,输入的参数是ok,通过sh-x进入shell调试模式,可以看出shell的执行细节和过程,因为if里面的条件满足,所以将signal置为1,接着再次进入while循环时,发现while条件为假,退出了while循环。
2.4.2 until循环语句以及应用举例
until命令和while命令类似,while能实现的脚本until同样也可以实现,但区别是while循环在条件为真时执行,而until循环则在条件为假时执行。
1.until循环的语法
下面是until循环的语法结构:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/97_01.jpg?sign=1739663791-X0FfwgBOzb8jufcIM2SCoy1ML6EHQyBp-0-9fe605add799137385207f0b2351ac76)
在执行while循环时,只要是expr的退出状态为0,将一直执行循环体。而until循环的执行过程是:当expr的退出状态不为0时,循环体将一直执行下去,直到退出状态为0时退出循环。
2.until循环执行流程
until循环的执行流程如图2-7所示。
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/97_02.jpg?sign=1739663791-GdUKu44i1xdhCuEBldx51gkVFCZt1y99-0-cad1efb3c72357c5771cbc640c89bb1f)
图2-7 until循环执行流程
从图中很容易理解until的执行逻辑,即为:条件不成立,则执行循环体,条件成立,退出循环体。注意,每执行一次循环体后,都会返回继续进行条件判断,如果为假,则继续执行循环体,否则,退出循环体。
3.until循环举例
下面通过例子来理解until循环的功能,脚本until.sh内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/97_03.jpg?sign=1739663791-3k6IZWIcDVX8UwfZifnulKBCJrSO4eL0-0-27ccede6b6b3d8a3eef5620167a69915)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/98_01.jpg?sign=1739663791-nSEbw2iCsBcdJNtu9tZGdQlzLTVB84lh-0-57c12eefe6ee28e71fb0851fc574d7dc)
这个脚本的功能是测试输入的IP是否能够ping通,如果能ping通,则进行ssh连接,无法ping通的话,等待60s,继续尝试ping,直到能ping通退出until循环。
执行这个脚本,输出如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/98_02.jpg?sign=1739663791-MJFs0YI3BuCSWGkbERRiWka5ifhXlO6N-0-10cc0788030cb0f31b9422f4a7f69a17)
第1次执行until.sh脚本时,能ping通给出的这个IP,所以直接退出until循环,进入done后面的ssh连接操作。
第2次执行until.sh脚本时,给出的IP地址无法ping通,所以就执行until循环体里面的命令,休眠60s,然后继续ping这个IP,如此反复这样循环,直到能ping通这个IP才退出until循环。
2.4.3 exit和sleep的应用环境与方法
exit命令用于退出脚本或当前进程。使用方法如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/98_03.jpg?sign=1739663791-f2fJyoC7qHPFdAg2CYDaYU3oi3AnBXRO-0-076926eed6673a3f96b0aa8ee695d71b)
n是一个从0~255的整数,0表示成功退出,非零表示遇到某种失败而非正常退出。该整数被保存在状态变量$?中。
有时候写shell的脚本,要顺序执行一系列的程序。有些程序在停止之后并不能立即退出,例如,有一个Tomcat出了问题,使用kill命令是不会瞬间结束掉的。而如果shell脚本还没等Tomcat彻底关闭,就接着执行下一行操作,那么shell的执行逻辑肯定就出问题了。如何解决这个问题呢?sleep命令可以实现休眠若干秒、若干分钟、若干小时。
sleep主要用来实现休眠指定的时间后,再去执行下面的命令,sleep命令用法如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/99_01.jpg?sign=1739663791-JY5NX4fzBubbiAnDuOYNGJH1EgO0Pipe-0-738121f86856efc6293a6752f873f0f8)
表示暂停ns,还可以指定具体的时间,例如,sleep 1表示暂停1s,sleep 1s也表示暂停1s,sleep 1m表示暂停1min,sleep 1h表示暂停1h。
下面来看个例子,脚本until1.sh内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/99_02.jpg?sign=1739663791-xZt6Hjw2wNMI5shhrPvAPwultUbd1dpo-0-3ef5e0a6f3c1b0ec0453139d00d3ab46)
这个脚本中,首先定义了一个username变量,这个变量是通过输入值来获取的。$1就是获取输入的第1个参数,如果没有输入参数的话,还会有输入提示,这是通过if判断的$#-lt 1条件实现的,$#表示命令行上参数的个数,也就是命令行参数个数如果小于1,那么就执行if里面的echo操作。
紧接着,下面是第2个if判断,这个判断用来获取输入的用户是否是系统里面的用户。如果是系统里面的用户,什么也不做,注意then后面的:,表示什么也不做,否则,给出提示,直接exit 2退出,注意,这个退出是退出shell脚本,后面的until循环不会再去执行。
最后一部分就是until的判断。until判断who|grep "$username"的返回状态,如果是非0,那么执行循环体里面的命令,否则退出循环体。
执行这个脚本,输出如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/99_03.jpg?sign=1739663791-0zOmj0hKjTm4QvoyhaB3y0isCBjkpiRJ-0-bef48237c5bc676cb535e3bb098d5c57)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/100_01.jpg?sign=1739663791-BzwqHPh4fXOML6s1thdtffWWoi5H75Bc-0-a6864b927492c9ed423d8747b064f10b)
上面执行这个脚本分了4种情况,第1种没有输入参数,就给出错误提示了;第2种,给出一个用户,然后判断发现ixdba是系统里面的用户,但是目前没有登录系统;第3种情况给出了一个非系统用户,然后就给出了非系统用户的提示,until循环并没有执行,这是因为shell中执行了exit命令,直接退出了shell脚本。从echo$?输出的返回状态码为2可得知,这是exit命令返回的值。最后一种情况是给定一个root用户,发现没任何提示,这是因为root用户是系统用户,并且也处于在线状态,所以什么都不输出。
2.4.4 select循环与菜单应用
一般地,使用while循环配合case可以实现循环与菜单功能,不过Bash提供了专门的select循环。select语法结构如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/100_02.jpg?sign=1739663791-gqZzneH887xwTlPUThPDxhOyHFIb2wui-0-55f7d18b97c4ff3827df1d227c8fe29f)
select循环主要用于创建菜单,按数字顺序排列的菜单项将显示在标准错误输出上,等待用户输入,菜单项的间隔符由环境变量IFS决定,用于引导用户输入的提示信息存放在环境变量PS3中,用户直接输入回车将重新显示菜单,与for循环类似,省略in list时等价于in“$*”。用户一旦输入菜单中的某个数字,则执行相应菜单中的命令。用户输入的内容被保存在内置变量REPLY中。
先看第1个例子,select1.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/100_03.jpg?sign=1739663791-zNfSOBjRdkVMeZzxqifHukLihFOXLGLk-0-1b19720540158234816b65050107d098)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/101_01.jpg?sign=1739663791-kB3hMBQsSGri5BAn2dyiAofN0RguLqpb-0-3cf389ec175ddf9309f69be5610705e6)
这是一个select里面嵌套case的例子,其中s是select里面的变量,有5个值可以选,分别是bash、perl、python、ruby和quit,对应数字为1~5。接着在select循环体中,执行case判断,注意,select是个无限循环,要退出循环,只能在循环体内用break命令退出循环或用exit命令终止脚本。执行这个脚本,结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/101_02.jpg?sign=1739663791-ABgS3i01y3PKGgrThPChMdAUdfUz6MJ7-0-f502b4e69d03494278b561878e10c717)
再看第2个例子,select2.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/101_03.jpg?sign=1739663791-G2I2fZ5m49mdLM7yVCc3uyJvyMeKjzgk-0-c3fa6fc10772df443af8c54da9a4acf0)
这个脚本中,select的变量是通过变量os和指定的分隔符确定的,其中1|2|3|4表示前面4个变量,输入其他值则退出循环体。执行脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/101_04.jpg?sign=1739663791-fdVcKoLWYMA8cJm9gyr7IWm8TZGhMHS8-0-5db5839f08584359f91513b3f89d26fc)
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/102_01.jpg?sign=1739663791-TYzpiyuZvIABp71bndGWTIyZ9iXni9gX-0-145965ca48bfa349bdc731528f833f2b)
最后再来看一个例子,select3.sh脚本内容如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/102_02.jpg?sign=1739663791-ehYJsZ8RMAaTvKMfHvu8iJNGwMCMKSm4-0-64baca17ab49027c3dd2911412a05040)
这个脚本用来判断指定的软件包是否安装,如果安装显示软件包名称,如果没有安装,则显示软件包没有安装的提示。执行脚本,输出结果如下:
![](https://epubservercos.yuewen.com/680E1C/17977546608668706/epubprivate/OEBPS/Images/102_03.jpg?sign=1739663791-LaGKm7RQIOFwjA8tNfkFsU24cULotjoJ-0-c5f802a6d2c6e83c82874ad6e1c497da)
从输出可以看出,当输入对应数字时,就会执行数字对应的内容检查,并给出检查结果。