您现在已经足够了解创建自己的循环了。在对循环感到满意之前,请在要处理的文件副本上使用它们,并尽可能多地使用带有内置保护措施的命令,以防止您破坏数据并造成不可弥补的错误,例如意外重命名整个文件,相同名称的文件目录,彼此覆盖。
有关高级for循环主题,请继续阅读。
并非所有的shell都是Bash
for关键字内置在Bash shell中。许多相似的shell使用相同的关键字和语法,但是某些shell(例如tcsh)使用不同的关键字(例如)来代替。
在tcsh中,语法本质上相似,但比Bash严格。在以下代码示例中,是否不键入字符串?在第2行和第3行中。它是辅助提示,提醒您仍在构建循环的过程中。
$ foreach f (*)
foreach? file $f
foreach? end
cat.jpg: JPEG image data, EXIF standard 2.2
design_maori.png: PNG image data, 4608 x 2592, 8-bit/color RGB, non-interlaced
otago.jpg: JPEG image data, EXIF standard 2.2
waterfall.png: PNG image data, 4608 x 2592, 8-bit/color RGB, non-interlaced
在tcsh中,和end都必须单独出现在单独的行中,因此不能像使用Bash和类似的shell那样在一行上创建for循环。
使用find命令执行for循环
从理论上讲,您可能会发现一个不提供for循环函数的shell,或者您可能只是更喜欢使用带有附加功能的其他命令。
find命令是实现for循环功能的另一种方法,因为它提供了几种方法来定义要包含在循环中的文件范围以及并行处理选项。
find命令旨在帮助您在硬盘驱动器上查找文件。它的语法很简单:您提供要搜索的位置的路径,并找到所有文件和目录:
$ find .
.
./cat.jpg
./design_maori.png
./otago.jpg
./waterfall.png
你可以通过添加name的一部分来过滤搜索结果:
$ find . -name "*jpg"
./cat.jpg
./otago.jpg
find的优点在于,可以使用-exec标志将找到的每个文件输入到循环中。例如,要仅缩小示例目录中的PNG照片,请执行以下操作:
$ find . -name "*png" -exec convert {} -scale 33% tmp/{} ;
$ ls -m tmp
design_maori.png, waterfall.png
在-exec子句中,括号字符{}代表正在处理的任何项(换句话说,已定位的任何以PNG结尾的文件,一次一个)。-exec子句必须以分号终止,但是Bash通常尝试自行使用分号。使用反斜杠(;)“转义”分号,以便find知道将分号视为其终止字符。
find命令非常擅长于其功能,有时它可能太好了。例如,如果重复使用它来查找另一个照片处理的PNG文件,则会出现一些错误:
$ find . -name "*png" -exec convert {} -flip -flop tmp/{} ;
convert: unable to open image `tmp/./tmp/design_maori.png':
No such file or directory @ error/blob.c/OpenBlob/2643.
...
似乎find找到了所有的PNG文件-不仅是当前目录(。)中的文件,还包括您之前处理过并放在tmp子目录中的文件。在某些情况下,您可能想要搜索当前目录以及其中的所有其他目录(以及其中的所有目录)。它可以是功能强大的递归处理工具,尤其是在复杂的文件结构中(例如,音乐艺术家的目录中包含充满音乐文件的专辑目录),但是您可以使用-选项对其进行限制。
只查找当前目录下的PNG文件(不包括子目录):
$ find . -maxdepth 1 -name "*png"
要在当前目录以及其他子目录级别中查找和处理文件,请将最大深度增加1:
$ find . -maxdepth 2 -name "*png"
它的默认值是进入所有子目录。
小延伸
使用循环的次数越多,节省的时间和精力就越多,可以处理的任务也就越大。您只是一个用户,但是经过深思熟虑的循环,您可以使计算机完成艰苦的工作。
您可以并且应该像对待其他任何命令一样对待循环,以便在需要对多个文件重复执行一个或两个操作时可以将其放在手边。但是,它也是进行认真编程的合法途径,因此,如果您必须对任意数量的文件执行复杂的任务,请抽出一些时间来计划工作流程。如果您可以在一个文件上实现目标,那么将该可重复过程包装在for循环中是相对简单的,并且唯一需要的“编程”是了解变量的工作方式以及足够的组织以将未处理的文件与已处理的文件分开。只需做一些练习,您就可以从一个Linux用户转移到知道如何编写循环的Linux用户!
Shell脚本关于循环的一些总结
不管是哪一门计算机语言,循环都是不可绕开的一个话题,Shell 当然也不是例外。下面总结一些 Shell 脚本里常用的循环相关的知识点,新手朋友可以参考。
for 循环
Shell 脚本里最简单的循环当属for循环,有编程基础的朋友应该都有使用过 for 循环。最简单的 for 循环如下所示,你只需将变量值依次写在 in 后面即可:
#!/bin/bash
for num in 1 2 3 4
do
echo $num
done
如果要循环的内容是字母表里的连续字母或连续数字,那么就可以按以下语法来写脚本:
#!/bin/bash
for x in {a..z}
do
echo $x
done
while 循环
除了 for 循环,Shell 同样提供了while循环。对于其它语言,如果你见过 for 循环却没见过 while 循环,那么你一定是学了个假语言。
在 while 循环里,每进行一次循环,条件都会被判断一次,来确定本次循环是否该继续。其实在循环次数比较少的情况下,for 循环与 while 循环效果差不多,但如果循环次数比较多,比如 10 万次,那么 while 循环的优势就体现出来了。
#!/bin/bash
n=1
while [ $n -le 4 ]
do
echo $n
((n++))
done
循环套循环
像其它高级语言一样,循环是可以互相嵌套的。比如下面这个例子,我们在 while 循环里再套入一个 for 循环:
#!/bin/bash
n=1
while [ $n -lt 6 ]
do
for l in {a..d}
do
echo $n$l
done
((n++))
done
这个脚本执行的结果应该是 1a, 1b, 1c, 1d, 2a, 2b … 5d。
循环的内容是变化的
我们上面提到的 for 循环,循环变量要赋的值都列在了 in 后面的列表里了。但这样灵活性太差,因为在很多情况下,循环变量要获得的值是不固定的。
就比如,有个变量要获得当前系统上所有用户,但因为每台电脑用户都不一样,我们根本就没办法将这个变量写死。
在这种情况下,我们可以使用 ls 命令将 /home 目录下所有用户都列出来,然后用循环变量依次获取它们。完整代码如下:
#!/bin/bash
for user in `ls /home`
do
echo $user
done
当然,除了 ls ,Shell 还支持其它命令。比如我们可以使用 date 命令获取当前系统时间,再依次打印出来:
$ for word in `date`
> do
> echo $word
> done
Thu
Apr
9
08:12:09
CST
2020
变量值检查
我们在使用 while 循环时,经常需要判断一个变量的值是否大于或者小于某个数。有时候这个数也是用另一个变量来表示,那么我们就需要判断这个变量的值是否是数字。有三种判断方法:
#!/bin/bash
echo -n "How many times should I say hello? "
read ans
if [ "$ans" -eq "$ans" ]; then
echo ok1
fi
if [[ $ans = *[[:digit:]]* ]]; then
echo ok2
fi
if [[ "$ans" =~ ^[0-9]+$ ]]; then
echo ok3
fi
第一种方法看起来似乎是个废话,但实际上,-eq只能用于数值间判断,如果是字符串则判断不通过,所以这就保证了 ans 是个数值型变量。
第二种方法是直接使用 Shell 的通配符对变量进行判断。
第三种方法就更直接了,使用正则表达式对变量进行判断。
我们直接来看一个例子:
#!/bin/bash
echo -n "How many times should I say hello? "
read ans
if [ "$ans" -eq "$ans" ]; then
n=1
while [ $n -le $ans ]
do
echo hello
((n++))
done
fi
在这个脚本里,我将要循环的次数传入到 ans 变量,然后脚本就具体打印几次 hello 。为了保证我们传入的内容是数字,我们使用了if [ “$ans” -eq “$ans” ]语句来判断。如果我们传入的不是数字,则不会进入 while 循环。
循环输出文本文件内容
如果你想按行依次循环输出文本文件的内容,可以这样操作:
#!/bin/bash
echo -n "File> "
read file
n=0
while read line; do
((n++))
echo "$n: $line"
done < $file
在这里,我们使用 read 命令将文本文件的内容读取存入 file 变量,然后再使用重定向(上述脚本最后一行)将 file 内容依次传入 while 循环处理再打印出来。
死循环
有时候我们需要一直永远循环做某件事,那么我们就可以使用死循环。达到这个目的很简单,只需使用while true即可。
#!/bin/bash
while true
do
echo -n "Still running at "
date
sleep 1
done
在以上这个脚本里,将每隔 1 秒打印一次Still at 具体时间,直到你按 Ctrl + C 终止这个脚本。
END