更多结构化命令
for 命令
bash shell 提供了 for 命令,允许你创建一个遍历一系列值的循环。每次迭代都使用其中一个 值来执行已定义好的一组命令。下面是 bash shell 中 for 命令的基本格式。
for var in list
do
commands
done
在 list 参数中,你需要提供迭代中要用到的一系列值。可以通过几种不同的方法指定列表 中的值。
在每次迭代中,变量 var 会包含列表中的当前值。第一次迭代会使用列表中的第一个值,第 二次迭代使用第二个值,以此类推,直到列表中的所有值都过一遍。
在 do 和 done 语句之间输入的命令可以是一条或多条标准的 bash shell 命令。在这些命令中, $var 变量包含着这次迭代对应的当前列表项中的值。
在
只要你愿意,也可以将 do 语句和 for 语句放在同一行,但必须用分号将其同列表中的值分 开:for var in list; do。
读取列表中的值
for 命令基本的用法就是遍历 for 命令自身所定义的一系列值。
...
for test in Alabama Alaska Arizona Arkansas California Colorado
do
echo The next state is $test
done
...
每次 for 命令遍历值列表,它都会将列表中的下个值赋给 t e s t 变 量 。 test变量。 test变量。test 变量可以像 for 命令语句中的其他脚本变量一样使用。在后一次迭代后,$test 变量的值会在 shell 脚本的剩余 部分一直保持有效。它会一直保持后一次迭代的值(除非你修改了它)。
读取列表中的复杂值
事情并不会总像你在 for 循环中看到的那么简单。有时会遇到难处理的数据。下面是给 shell 脚本程序员带来麻烦的典型例子。
for test in I don't know if this'll work
do
echo "word:$test"
done
shell 看到了列表值中的单引号并尝试使用它们来定义一个单独的数据值,这真是把 事情搞得一团糟。
有两种办法可解决这个问题:
- 使用转义字符(反斜线)来将单引号转义;
- 使用双引号来定义用到单引号的值。
for test in I don\'t know if "this'll" work
do
echo "word:$test"
done
在第一个有问题的地方添加了反斜线字符来转义 don’t 中的单引号。在第二个有问题的地方 将 this’ll 用双引号圈起来。两种方法都能正常辨别出这个值。
你可能遇到的另一个问题是有多个词的值。记住,for 循环假定每个值都是用空格分割的。 如果有包含空格的数据值,你就陷入麻烦了
如果在单独的数据值中有 空格,就必须用双引号将这些值圈起来。
从变量读取列表
通常 shell 脚本遇到的情况是,你将一系列值都集中存储在了一个变量中,然后需要遍历变量 中的整个列表。也可以通过 for 命令完成这个任务。
list="Alabama Alaska Arizona Arkansas Colorado"
list=$list" Connecticut"
for state in $list
do
echo "Have you ever visited $state?"
done
$list 变量包含了用于迭代的标准文本值列表。注意,代码还是用了另一个赋值语句向$list 变量包含的已有列表中添加(或者说是拼接)了一个值。这是向变量中存储的已有文本字符串尾 部添加文本的一个常用方法。
从命令读取值
生成列表中所需值的另外一个途径就是使用命令的输出。可以用命令替换来执行任何能产生 输出的命令,然后在 for 命令中使用该命令的输出。
file="states"
for state in $(cat $file)
do
echo "Visit beautiful $state"
done
这个例子在命令替换中使用了 cat 命令来输出文件 states 的内容。你会注意到 states 文件中每一 行有一个州,而不是通过空格分隔的。for 命令仍然以每次一行的方式遍历了 cat 命令的输出, 假定每个州都是在单独的一行上。但这并没有解决数据中有空格的问题。如果你列出了一个名字 中有空格的州,for 命令仍然会将每个单词当作单独的值。
更改字段分隔符
IFS 环境变量定义了 bash shell 用作字段分隔符的一系列字符。默认情况下,bash shell 会将下列字 符当作字段分隔符:
- 空格
- 制表符
- 换行符
如果 bash shell 在数据中看到了这些字符中的任意一个,它就会假定这表明了列表中一个新数 据字段的开始。在处理可能含有空格的数据(比如文件名)时,这会非常麻烦
要解决这个问题,可以在 shell 脚本中临时更改 IFS 环境变量的值来限制被 bash shell 当作字段 分隔符的字符。例如,如果你想修改 IFS 的值,使其只能识别换行符,那就必须这么做:
IFS=$'\n'
将这个语句加入到脚本中,告诉 bash shell 在数据值中忽略空格和制表符。
在处理代码量较大的脚本时,可能在一个地方需要修改 IFS 的值,然后忽略这次修改,在 脚本的其他地方继续沿用 IFS 的默认值。一个可参考的安全实践是在改变 IFS 之前保存原 来的 IFS 值,之后再恢复它。 这种技术可以这样实现:
IFS.OLD=$IFS IFS=$'\n' <在代码中使用新的IFS值> I