本系列(玩转Bash脚本)更多文章,

 

和其他语言一样Bash的循环结构中也有while语句。



基本结构




1 while 条件
2 do
3 循环体
4 done


 

和for语句一样,它的循环体同样是do…done结构。我们可以把while语句再折叠一下




1 while 条件;do
2 循环体
3 done


 

还能进一步折叠成一行体




1 while 条件;do 循环体;done


 

Bash语句大都可以写作一行,只不过可读性差

和其他编程语言不同的是,Bash中的while语句用法是比较多样的。可以根据while条件的不同,将while语句分为几类。



while条件



方括号[ ]

和if语句的条件相同。即操作符​​[ ]​​。关于​​[ ]​​的用法请参考之前的文章。




1 #!/bin/bash
2 n=1
3 while [ $n -le 10 ]
4 do
5 echo $n
6 let n++ #或者写作n=$(( $n + 1 ))
7 done


 

在前面的文章中,已经讲过了操作符​​[ ]​​和​​test​​语句是等价的,所以这里也可以用test语句来作条件。



终端命令

while的条件可以是各种终端的命令。包括外部命令或bash内建(built-in)命令都可以。因为命令都是有返回值的(可以用echo $?查看),命令执行的成功与否就是while条件的真或假。

以read命令来举个例子




1 #!/bin/bash
2 while read var;do
3 echo "您输入的是$var"
4 done


 

这个程序是个死循环,将不停地等待您的输入,并回显出来。

这里的命令可以是单个命令也可以是组合命令,比如用逻辑连接符连接的命令,或者管道、重定向组成的长命令



死循环

除了让while条件恒成立外,编程语言都有一种简洁的死循环写法。比如C语言中典型的死循环条件是while(1),而java中的写法是while(true)。 

而Bash中的写法则简单的多,只需要一个冒号。




1 #!/bin/bash
2 while :
3 do
4 echo I love you forever
5 done


 

这是一个死循环,执行之后请用按组合键Ctrl+C来终止它。 

此外,还有一种死循环写法就是利用系统自带的true命令(/bin/true)




1 #!/bin/bash
2 while /bin/true
3 do
4 echo I love you forever
5 done


 

由于我们的系统环境变量(PATH)中一般都包含了路径/bin,所有我们也可以简写成​​while true​



while实现菜单demo

我们或许曾经用C/C++在控制台上输出过菜单。这通常是一个do-while循环实现的,先输出菜单的每个选项,然后等待输入, 

根据不同的输入执行不同的操作,然后循环再次输出菜单……。 

bash中没有do-while风格的循环,但是我们很容易用替代的方案实现该功能。用死循环+if/case条件判断语句就够了。




1 #!/bin/bash
2 #菜单demo
3 while :
4 do
5 echo #输出空行
6 echo "========================="
7 echo " 1:输出成绩单"
8 echo " 2:输出课程表"
9 echo " 3:输出空闲教室"
10 echo " q:退出菜单"
11 echo "========================="
12 read -p"请输入:" input
13 case $input in
14 1)echo "稍等,正在为您输出成绩单";;
15 2)echo "稍等,正在为您输出课程表";;
16 3)echo "稍等,正在为您输出空闲教";;
17 q|Q) exit
18 esac
19 done


 



while与重定向

while语句可以联合重定向(>和<)一起使用。



while和输入重定向<

格式为




1 while 命令
2 do
3 循环体
4 done < 文件名


 

相当于将文件内容逐行传递给while后面的命令(类似管道),然后再执行循环体。 

当循环体为空的时候,这个结构的功能通常和cat没有两样,比如:




1 while grep "love"
2 do
3 done < letter


 

从文件letter中查找love这个单词,其实和cat letter|grep “love”没什么两样。

这个结构更习惯的用法和read联用,来将文件内容逐行取出,赋值给read后面的变量。比如:




1 #!/bin/bash
2 #从/etc/passwd文件中读取用户名并输出
3 oldIFS=$IFS #IFS是文件内部分隔符
4 IFS=":" #设置分隔符为:
5 while read username var #var变量不可少
6 do
7 echo "用户名:$username"
8 done < /etc/passwd
9 IFS=$oldIFS


 

熟悉/etc/passwd文件结构的朋友都知道这个文件的每一行包含了一个用户的大量信息(用户名只是第一项)。 

这里我们实际只输出了用户名。但是注意while的read后面除变量username外还有个var,尽管我们并不输出这个变量的值。 

但它却必不可少,如果我们写成​​while read username​​那么username的值等于passwd文件这一整行的内容(IFS=”:”也就不起作用了)

bash中,只有当有多个变量要从一行文本赋值的时候,才尝试去用IFS来分割,然后赋值。



while和输出重定向>

格式为




1 while 命令
2 do
3 循环体
4 done > 文件名


 

这个结构会将命令的输出,以及循环体中的标准输出都重定向到指定的文件中。 

比如:




1 #!/bin/bash
2 #每隔10分钟,ping一下局域网内主机192.168.1.101,
3 #并把结果记录到ping.txt文件中
4 while date
5 do
6 ping -c5 192.168.1.101 >/dev/null 2>&1
7 if [ $? = 0 ];then
8 echo OK
9 else
10 echo FAIL
11 fi
12 sleep 600 #600秒是10分钟
13 done > ping.txt


 




我们来cat一下ping.txt文件,看一看:



2015年 01月 31日 星期六 16:03:13 CST
OK