上次已经学习了基础的结构化命令if语句,循环语句也是使用频率非常高的结构化命令,比如for语句、while语句等。一起来学习吧~
要点一、for命令
1、基础格式
#!/bin/bash
for test in one two three four five
do
echo the number is $test
done
如上关于for语句的格式:
-
test
为变量; -
in
后面的系列值为list参数,以空格隔开,变量依次迭代;
在循环结束后,$test变量会在脚本中保持最后一个值(five)。当然,除非你修改了它。
- 在 do 和 done 之间为一/多条标准的bash shell命令;
- 关于写法,for 与 do 可以放在一行里,用分号
;
隔开。如下
for test in one two three four five; do
echo the number is $test
done
2、关于list参数
上述举的例子应该是最简单的形式:用空格分隔的文本字符串。针对更加实用的情况,list参数也可以是以下的形式
2.1、储存变量读取
#!/bin/bash
list="one two three four five"
for test in $list
do
echo the number is $test
done
- 如上代码,将系列值储存在list变量里,供test迭代。
2.2、文本文件/命令替换读取
cat number.txt
one
two
three
four
five
- 如上,使用
cat
命令准备文本文件
#!/bin/bash
file="number.txt" #将文件引入脚本里的变量
for test in $(cat $file)
do
echo the number is $test
done
- 使用命令替换符
$( )
读取
2.3、字段分隔符
内部字段分隔符(internal field separator)环境变量为IFS,其默认的分隔符有空格、制表符、换行符。所以上述例子中有采用空格分隔的;有采用换行符分隔的。
- 但是可能会遇到以下情况:
(1)每行文本里有空格符,但只想根据换行符分隔,需要重新定义默认分隔符。
cat number1.txt
one two
three
four
five six
#!/bin/bash
IFS=$'\n' #设定分隔符仅为换行符,忽略空格
file="number1.txt"
for test in $(cat $file)
do
echo the number is $test
done
针对上述情况,可以将含有空格的行文字加上双引号,而不用修改IFS。这种解决办法也适用于文本中本身含有引号的干扰情况。
(2)在脚本里修改环境变量IFS仅会在脚本中生效,不影响外部环境。如果在一个脚本中需要在一处更改默认分隔符,而后续需要沿用原来的默认设置。
IFS.OLD=$IFS
IFS=$'\n'
#
#
#
IFS=$IFS.OLD
(3)在某种情况下,需要指定多个新的分隔符
IFS=$'\n':;"
- 上述代码分别指定换行符、冒号、分号,双引号作为字段分隔符。
此外关于list参数,还有一个技巧:通配符。比如
for file in /home/li/l*
代码能够遍历/home/li/目录中所有以l开头的对象。
3、C语言风格的for命令
#!/bin/bash
for (( a = 1; a < 10; a++ ))
do
echo the number is $a
done
记住第一行特殊格式,赋值可以有空格,条件中的变量不是以美元符开头。
- 第一部分为原始值;
- 第二部分为循环条件;
- 第三部分为迭代过程(递增/递减)。
要点二、while命令
#!/bin/bash
var1=10
while [ $var1 -gt 0 ]
do
echo $var1
var1=$[ $var1 - 1 ]
done
关于while语句的注意点有:
- while后面的语句判断后面的命令是命令退出状态码,若返回0(成功执行),则继续do的命令;
- do后面的命令一般包括两部分。一是执行目标命令(循环的最终目的)
echo $var1
、二是迭代变量值,var1=$[ $var1 - 1 ]
再次返回给while 进行判断。直到while后的退出状态码不为0 - done 标志循环语句结束。
- 值得注意的是在 while后可执行目标命令,但我觉得应该不常见。
此外还有until语句,与while相反就不探究了(详见p272)
要点三、嵌套循环
嵌套循环就是在迭代中再使用迭代,命令的执行次数为乘积关系。下面以两个简单例子了解一下~
for嵌套for
#!/bin/bash
for (( a = 1; a < 3; a++ ))
echo "Outer loop $a :"
for (( b = 1; b <= 3; b++ ))
do
echo " Inside loop: $b"
done
done
while嵌套for
#!/bin/bash
#
var1=5
#
while [ $var1 -ge 0 ]
do
echo "Outer loop: $var1"
for (( var2 = 1; var2 < 3; var2++ ))
do
var3=$[ $var1*$var2 ]
echo " Inner loop: $var1 * $var2 = $var3"
done
var1=$[ $var1 - 1 ] #在最后交代while变量的迭代
done
要点四、控制循环
在循环条件下,有时并非list参数所举的全部系列值都想采用,也存在只取部分值的情况。此时需要使用控制循环命令,一般搭配if语句,表明终止循环的条件。(我觉得也可以直接修改list参数的值。针对具体情况,怎么方便怎么来吧。)
break命令
一旦执行,直接终止退出,不在考虑后面的list参数。
#!/bin/bash
for var1 in 1 2 3 4 5 6 7 8 9
do
if [ $var1 -eq 5 ]
then
break
fi
echo "Iteration number: $var1"
done
echo "The for loop is completed"
上述代码结果中,var1会变到4;到5时,因满足if条件,执行break
命令而终止循环
continue命令
- 不会直接完全终止,可以设置终止的list参数范围
#!/bin/bash
for (( var1 = 1; var1 < 15; var1++ ))
do
if [ $var1 -gt 5 ] && [ $var1 -lt 10 ]
then
continue
fi
echo "Iteration number: $var1"
done
- var1值为1时,退出码不为0;不执行then的continue命令,往下执行echo语句;
- 若为6,则执行then语句,跳过这一圈剩余的循环命令,继续下一个for取值7;
- 取到10时,退出码不为0;不执行then的continue命令,往下执行echo语句。
关于循环控制,也可在嵌套循环里执行
break
或continue
。因为涉及到内循环、外循环,以及控制命令加不加参数的变化,不在赘述了,详见p277。
最后以一个实例结束
- 目的:在PATH环境变量中,查找系统有哪些可执行文件可供使用。
#!/bin/bash
IFS=:
for folder in $PATH
do
echo "$folder:"
for file in $folder/*
do
if [ -x $file ]
then
echo " $file"
fi
done
done
./test.sh | more
补充一个知识点:在shell脚本中,可以对循环的输出使用管道或进行重定向;直接在done
命令后添加相关命令即可。如下:
done > output.txt
done | sort
参考教材为《Linux命令行与shell脚本编程大全》