1、for循环介绍
for
循环是固定循环,也就是在循环时已经知道需要进行几次的循环,有时也把for
循环称为计数循环。
在Shell中for
循环的语法有如下两种:
# 语法1: for 变量 in 值1 值2 值3 … do 程序 done
这种语法中for
循环的次数,取决于in
后面值的个数(空格分隔),有几个值就循环几次,并且每次循环都把该值赋予变量。也就是说,假设in
后面有三个值,for
会循环三次,第一次循环会把值1赋予变量,第二次循环会把值2赋予变量,以此类推。
# 语法2: for((初始值;循环控制条件;变量变化)) do 程序 done
语法二中需要注意:
- 初始值:在循环开始时,需要给某个变量赋予初始值,如
i=1
; - 循环控制条件:用于指定变量循环的次数,如
i<=100
,则只要i的值小于等于100,循环就会继续; - 变量变化:每次循环之后,变量该如何变化,如
i=i+1
,代表每次循环之后,变量i
的值都加1。
2、示例
语法一举例:
需求:打印时间。
# 创建脚本文件 [root@localhost ~]# vim sh/for.sh #!/bin/bash for time in morning noon afternoon evening do echo "This time is $time!" done
执行脚本结果:
[root@localhost tmp]# chmod 755 for1.sh [root@localhost tmp]# ./for1.sh This time is morning! This time is noon! This time is afternoon! This time is evening!
语法二举例:
语法二就和其他语言中的for
循环类似了,也就是事先决定循环次数的固定循环了。
需求:从1加到100。
#!/bin/bash # 定义一个求和变量sum sum=0 # 定义循环100次 # 在Shell中如果要进行数学运算,需要用双小括号括起来,才识别括号里面是数值运算。 for((i=1;i<=100;i=i+1)) do # 每次循环给变量sum赋值 sum=$(($sum+$i)) done # 输出1加到100的和 echo "The sum of 1+2+...+100 is :$sum"
3、for循环总结
- 第一种格式的
for
循环是最常见的Shell循环方式。 - 第二种格式的
for
循环适合做数学运算,可以方便的指定循环次数。
4、练习:批量解压缩脚本
方式一:批量解压缩
# 创建脚本文件auto-tar.sh [root@localhost ~]# vim sh/auto-tar.sh # 批量解压缩脚本 #!/bin/bash # 进入压缩包目录。 cd /tmp/sh/tar # 把tar目录中的所有压缩包的文件名,保存到tar.log文件中。 # 单>是覆盖。 # 而且tar.log中内容是每一个文件名是一行。 ls *.tar.gz>tar.log # 把tar目录中.tgz类型的压缩包的名字也追加到tar.log文件中。 # 双>>是追加。 ls *.tgz>>tar.log &>/dev/null # 提示:用上面的方式,把需要解压的所有类型的压缩文件的名称,都存入到tar.log文件中。 # 读取tar.log文件的内容,文件中有多少个值,就会循环多少次, # 每次循环把文件名赋予变量i for i in $(cat tar.log) do # 解压缩,并把所有输出都丢弃 tar -zxvf $i &>/dev/null # 注意如果还有其他格式的压缩包,需要在这里进行if判断, # 分别针对不同格式的压缩文件进行解压。 # 方式二也一样。 done # 删除临时文件tar.log,因为脚本执行完就没有作用了。 rm -rf /tmp/sh/tar/tar.log
说明:
第一种方式的for
循环,in
后有几个值,就循环几次,值之间要有空格分隔。
而tar.log
文件中存放的是6个压缩包的文件名,且每一个文件名占一行,
[root@localhost tmp]# cat tar.log apr-1.4.6.tar.gz apr-util-1.4.1.tar.gz httpd-2.4.7.tar.gz I mysq1-5.5.23.tar.gz php-5.6.15.tar.gz phpMyAdmin-4.1.4-al1-languages.tar.gz
这样的格式,就相当于一行算一个值,这样就可以循环6次,每次的值就是一个压缩包的文件名,
这样就完成了所需文件的批量解压缩。
方式二:批量解压缩
用for
循环的第二种方式进行批量解压缩,有两个需要注意的内容。
- 第一:需要知道压缩包的总个数,因为我需要用
for
循环的第二种格式进行批量解压,就需要先知道要循环几次。
解决方式:把所有需要解压文件的文件名保存到一个文件中(临时文件),这时候所需解压缩文件的文件名就变成了字符串,然后通过wc
命令进行统计就可以。 - 第二:需要把每个压缩包的名称提取出来,赋值在变量中。
就是第一次循环,变量中赋值的是第一个压缩包的文件名,第二次循环,变量中赋值第二个压缩包的文件名,然后就能够用tar
命令解压该压缩包了。
#/bin/bash # 进入压缩包目录。 cd /tmp/sh/tar # 把tar目录中的所有压缩包的文件名,保存到tar.log文件中。 # 单>是覆盖。 # 而且tar.log中内容是每一个文件名是一行。 ls *.tar.gz>tar.log # 把tar目录中.tgz类型的压缩包的名字也追加到tar.log文件中。 # 双>>是追加。 ls *.tgz>>tar.log &>/dev/null # 提示:用上面的方式,把需要解压的所有类型的压缩文件的名称,都存入到tar.log文件中。 # wc -l命令统计行号,也就是获取文件个数。 num=$(cat /tmp/sh/tar/tar.log | wc -l) #或者:wc -l /tmp/sh/tar/tar.log # 开始遍历解压文件 for((i=1;i<="$num";i=i+1)) do # 用awk命令提取文件名,来获取解压文件的文件名 # NR是awk的内置变量,表示当前awk所处理的行,是总数据的第几行。 # 注意'$i'这个地方,依然要使用单引号,使用双引号会报错。 # awk 'NR=='$i' {print $1} 意思是获取第几行的第几列信息。 filename=$(cat tar.log | awk 'NR=='$i' {print $1}) # 解压文件 tar -zxvf $filename -C /tmp/sh/tar done # 删除临时文件tar.log rm -rf /tmp/sh/tar/tar.log
总结:
for
循环的第一种方式,适合作为Shell脚本的编写,更为简单。