第34章 陷阱

Turandot: Gli enigmi sono tre, la morte una!
Caleph: No, no! Gli enigmi sono tre, una la vita!

——Puccini

以下的做法(非推荐)将让你原本平淡无奇的生活激动不已。

  • 将保留字或特殊字符声明为变量名。
  1. case=value0 # 引发错误。
  2. 23skidoo=value1 # 也会引发错误。
  3. # 以数字开头的变量名是被shell保留使用的。
  4. # 试试_23skidoo=value1。以下划线开头的变量名就没问题.
  5. # 然而 . . . 只用一个下划线作为变量名就不行。
  6. _=25
  7. echo $_ # $_是一个特殊变量, 代表最后一个命令的最后一个参数。
  8. # 但是,_是一个有效的函数名!
  9. xyz((!*=value2 # 引起严重的错误。
  10. # Bash3.0之后,标点不能出现在变量名中。
  • 使用连字符或其他保留字符来做变量名(或函数名)。
  1. var-1=23
  2. # 用 'var_1 代替。
  3. function-whatever () # 错误
  4. # 用 ‘function_whatever ()’ 代替。
  5. # Bash3.0之后,标点不能出现在函数名中。
  6. function.whatever () # 错误
  7. # 用 ‘functionWhatever ()’ 代替。
  • 让变量名与函数名相同。 这会使得脚本的可读性变得很差。
  1. do_something ()
  2. {
  3. echo "This function does something with \"$1\"."
  4. }
  5. do_something=do_something
  6. do_something do_something
  7. # 这么做是合法的,但会让人混淆。
  • 不合时宜的使用空白符。与其他编程语言相比,Bash非常讲究空白符的使用。
  1. var1 = 23 # ‘var1=23’才是正确的。
  2. # 对于上边这一行来说,Bash会把“var1”当作命令来执行,
  3. # “=”和“23”会被看作“命令”“var1”的参数。
  4. let c = $a - $b # ‘let c=$a-$b’或‘let "c = $a - $b"’才是正确的。
  5. if [ $a -le 5] # if [ $a -le 5 ] 是正确的。
  6. # ^^ if [ "$a" -le 5 ] 这么写更好。
  7. # [[ $a -le 5 ]] 也行。
  1. { ls -l; df; echo "Done." }
  2. # bash: syntax error: unexpected end of file
  3. { ls -l; df; echo "Done."; }
  4. # ^ ### 最后的这条命令必须以分号结尾。
  • 假定未被初始化的变量(赋值前的变量)被“清0”。事实上,未初始化的变量值为“null”,而不是0。
  1. #!/bin/bash
  2. echo "uninitialized_var = $uninitialized_var"
  3. # uninitialized_var =
  4. # 但是 . . .
  5. # if $BASH_VERSION ≥ 4.2; then
  6. if [[ ! -v uninitialized_var ]]
  7. then
  8. uninitialized_var=0 # Initialize it to zero!
  9. fi
  • 混淆测试符号=和-ep。请记住,=用于比较字符变量,而-ep用来比较整数。
  1. if [ "$a" = 273 ] # $a是整数还是字符串?
  2. if [ "$a" -eq 273 ] # $a为整数。
  3. # 有些情况下,即使你混用-ep和=,也不会产生错误的结果。
  4. # 然而 . . .
  5. a=273.0 # 不是一个整数。
  6. if [ "$a" = 273 ]
  7. then
  8. echo "Comparison works."
  9. else
  10. echo "Comparison does not work."
  11. fi # Comparison does not work.
  12. # 与a=" 273"和a="0273"相同。
  13. # 类似的, 如果对非整数值使用“-ep”的话,就会产生问题。
  14. if [ "$a" -eq 273.0 ]
  15. then
  16. echo "a = $a"
  17. fi # 产生了错误消息而退出。
  18. # test.sh: [: 273.0: integer expression expected

样例 34-1. 数字比较与字符串比较并不相同

  1. #!/bin/bash
  2. # bad-op.sh: 尝试一下对整数使用字符串比较。
  3. echo
  4. number=1
  5. # 下面的"while循环"有两个过错误:
  6. #+ 一个比较明显,而另一个比较隐蔽。
  7. while [ "$number" < 5 ] # 错!应该是: while [ "$number" -lt 5 ]
  8. do
  9. echo -n "$number "
  10. let "number += 1"
  11. done
  12. # 如果试图运行这个错误的脚本,就会得到一个错误信息:
  13. #+ bad-op.sh: line 10: 5: No such file or directory
  14. # 在单中括号结构([ ])中,"<"必须被转义,
  15. #+ 即便如此,比较两个整数仍是错误的。
  16. echo "---------------------"
  17. while [ "$number" \< 5 ] # 1 2 3 4
  18. do #
  19. echo -n "$number " # 看起来好像可以工作,但是 . . .
  20. let "number += 1" #+ 事实上是比较ASCII码,
  21. done #+ 而不是整数比较。
  22. echo; echo "---------------------"
  23. # 这么做会产生问题。比如:
  24. lesser=5
  25. greater=105
  26. if [ "$greater" \< "$lesser" ]
  27. then
  28. echo "$greater is less than $lesser"
  29. fi # 105 is less than 5
  30. # 事实上,在字符串比较中(按照ASCII码的顺序)
  31. #+ "105"小于"5"。
  32. echo
  33. exit 0
  • 试图用let来设置字符串变量。
  1. let "a = hello, you"
  2. echo "$a" # 0
  • 有时候在“test”中括号([ ])结构里的变量需要被引用起来(双引号)。如果不这么做的话,可能会引起不可预料的结果。请参考例子 7-6例子 16-5例子 9-6

  • 为防分隔,用双引号引用一个包含空白符的变量。 有些情况下,这会产生意想不到的后果

  • 脚本中的命令可能会因为脚本宿主不具备相应的运行权限而导致运行失败。如果用户在命令行中不能调用这个命令的话,那么即使把它放到脚本中来运行,也还是会失败。这时可以通过修改命令的属性来解决这个问题,有时候甚至要给它设置suid位(当然, 要以root身份来设置)。

  • 试图使用-作为作为重定向操作符(事实上它不是),通常都会导致令人不快的结果。

  1. command1 2> - | command2
  2. # 试图将command1的错误输出重定向到一个管道中 . . .
  3. # . . . 不会工作。
  4. command1 2>& - | command2 # 也没效果。
  5. 感谢,S.C
  • 使用Bash 2.0或更高版本的功能,可以在产生错误信息的时候,引发修复动作。但是比较老的Linux机器默认安装的可能是Bash 1.XX。
  1. #!/bin/bash
  2. minimum_version=2
  3. # 因为Chet Ramey经常给Bash添加一些新的特征,
  4. # 所以你最好将$minimum_version设置为2.XX,3.XX,或是其他你认为比较合适的值。
  5. E_BAD_VERSION=80
  6. if [ "$BASH_VERSION" \< "$minimum_version" ]
  7. then
  8. echo "This script works only with Bash, version $minimum or greater."
  9. echo "Upgrade strongly recommended."
  10. exit $E_BAD_VERSION
  11. fi
  12. ...
  1. var=1 && ((--var)) && echo $var
  2. # ^^^^^^^^^ 在这里,这个与列表返回错误代码1而终止。
  3. # 不会打印$var的值!
  4. echo $? # 1
  • 一个带有DOS风格换行符(\r\n)的脚本将会运行失败,因为#!/bin/bash\r\n是不合法的,与我们所期望的#!/bin/bash\n不同,解决办法就是将这个脚本转换为UNIX风格的换行符。
  1. #!/bin/bash
  2. echo "Here"
  3. unix2dos $0 # 脚本先将自己改为DOS格式。
  4. chmod 755 $0 # 更改可执行权限。
  5. # 'unix2dos'会删除可执行权限
  6. ./$0 # 脚本尝试再次运行自己。
  7. # 但它作为一个DOS文件,已经不能运行了。
  8. echo "There"
  9. exit 0
  • #!/bin/sh开头的Bash脚本,不能在完整的Bash兼容模式下运行。某些Bash特定的功能可能会被禁用。如果脚本需要完整的访问所有Bash专有扩展,那么它需要使用#!/bin/bash作为开头。

  • 如果在here document中,结尾的limit string之前加上空白字符的话,将会导致脚本的异常行为。

  • 在一个输出被捕获的函数中放置了不止一个echo语句。

  1. add2 ()
  2. {
  3. echo "Whatever ... " # 删掉zhehan
  4. let "retval = $1 + $2"
  5. echo $retval
  6. }
  7. num1=12
  8. num2=43
  9. echo "Sum of $num1 and $num2 = $(add2 $num1 $num2)"
  10. # Sum of 12 and 43 = Whatever ...
  11. # 55
  12. # 这些echo连在一起了。

这是行不通的。

  • 脚本不能将变量export到它的父进程(即调用这个脚本的shell),或父进程的环境中。就好比我们在生物学中所学到的那样,子进程只会继承父进程, 反过来则不行。
  1. WHATEVER=/home/bozo
  2. export WHATEVER
  3. exit 0
  1. bash$ echo $WHATEVER
  2. bash$
  • 可以确定的是,即使回到命令行提示符,变量$WHATEVER仍然没有被设置。

  • 子shell中设置和操作变量之后,如果尝试在子shell作用域之外使用同名变量的话, 将会产生令人不快的结果。

样例 34-2. 子shell缺陷

  1. #!/bin/bash
  2. # 子shell中的变量缺陷。
  3. outer_variable=outer
  4. echo
  5. echo "outer_variable = $outer_variable"
  6. echo
  7. (
  8. # 开始子shell
  9. echo "outer_variable inside subshell = $outer_variable"
  10. inner_variable=inner # Set
  11. echo "inner_variable inside subshell = $inner_variable"
  12. outer_variable=inner # 会修改全局变量吗?
  13. echo "outer_variable inside subshell = $outer_variable"
  14. # 如果将变量‘导出’会产生不同的结果么?
  15. # export inner_variable
  16. # export outer_variable
  17. # 试试看。
  18. # 结束子shell
  19. )
  20. echo
  21. echo "inner_variable outside subshell = $inner_variable" # Unset.
  22. echo "outer_variable outside subshell = $outer_variable" # Unchanged.
  23. echo
  24. exit 0
  25. # 如果你去掉第19和第20行的注释会怎样?
  26. # 会产生不同的结果吗?
  • 将echo的输出通过管道传递给read命令可能会产生不可预料的结果。在这种情况下,read命令的行为就好像它在子shell中运行一样。可以使用set命令来代替(就好像例子15-18一样)。

样例 34-3. 将echo的输出通过管道传递给read命令

  1. #!/bin/bash
  2. # badread.sh:
  3. # 尝试使用'echo'和'read'命令
  4. #+ 非交互的给变量赋值。
  5. # shopt -s lastpipe
  6. a=aaa
  7. b=bbb
  8. c=ccc
  9. echo "one two three" | read a b c
  10. # 尝试重新给变量a,b,和c赋值。
  11. echo
  12. echo "a = $a" # a = aaa
  13. echo "b = $b" # b = bbb
  14. echo "c = $c" # c = ccc
  15. # 重新赋值失败。
  16. ### 但如果 . . .
  17. ## 去掉第6行的注释:
  18. # shopt -s lastpipe
  19. ##+ 就能解决这个问题!
  20. ### 这是Bash 4.2版本的新特性。
  21. # ------------------------------
  22. # 试试下边这种方法。
  23. var=`echo "one two three"`
  24. set -- $var
  25. a=$1; b=$2; c=$3
  26. echo "-------"
  27. echo "a = $a" # a = one
  28. echo "b = $b" # b = two
  29. echo "c = $c" # c = three
  30. # 重新赋值成功。
  31. # ------------------------------
  32. # 也请注意,echo到'read'的值只会在子shell中起作用。
  33. # 所以,变量的值*只*会在子shell中被修改。
  34. a=aaa # 重新开始。
  35. b=bbb
  36. c=ccc
  37. echo; echo
  38. echo "one two three" | ( read a b c;
  39. echo "Inside subshell: "; echo "a = $a"; echo "b = $b"; echo "c = $c" )
  40. # a = one
  41. # b = two
  42. # c = three
  43. echo "-----------------"
  44. echo "Outside subshell: "
  45. echo "a = $a" # a = aaa
  46. echo "b = $b" # b = bbb
  47. echo "c = $c" # c = ccc
  48. echo
  49. exit 0

事实上,也正如Anthony Richardson指出的那样,通过管道将输出传递到任何循环中, 都会引起类似的问题。

  1. # 循环的管道问题。
  2. # 这个例子由Anthony Richardson编写,
  3. #+ 由Wilbert Berendsen补遗。
  4. foundone=false
  5. find $HOME -type f -atime +30 -size 100k |
  6. while true
  7. do
  8. read f
  9. echo "$f is over 100KB and has not been accessed in over 30 days"
  10. echo "Consider moving the file to archives."
  11. foundone=true
  12. # ------------------------------------
  13. echo "Subshell level = $BASH_SUBSHELL"
  14. # Subshell level = 1
  15. # 没错, 现在是在子shell中运行。
  16. # ------------------------------------
  17. done
  18. # 变量foundone在这里肯定是false,
  19. #+ 因为它是在子shell中被设置为true的。
  20. if [ $foundone = false ]
  21. then
  22. echo "No files need archiving."
  23. fi
  24. # =====================现在,下边是正确的方法:=================
  25. foundone=false
  26. for f in $(find $HOME -type f -atime +30 -size 100k) # 这里没使用管道。
  27. do
  28. echo "$f is over 100KB and has not been accessed in over 30 days"
  29. echo "Consider moving the file to archives."
  30. foundone=true
  31. done
  32. if [ $foundone = false ]
  33. then
  34. echo "No files need archiving."
  35. fi
  36. # ==================这里是另一种方法==================
  37. # 将脚本中读取变量的部分放到一个代码块中,
  38. #+ 这样一来,它们就能在相同的子shell中共享了。
  39. # 感谢,W.B。
  40. find $HOME -type f -atime +30 -size 100k | {
  41. foundone=false
  42. while read f
  43. do
  44. echo "$f is over 100KB and has not been accessed in over 30 days"
  45. echo "Consider moving the file to archives."
  46. foundone=true
  47. done
  48. if ! $foundone
  49. then
  50. echo "No files need archiving."
  51. fi
  52. }
  • 一个相关的问题:当你尝试将tail -f的stdout通过管道传递给grep时,会产生问题。
  1. tail -f /var/log/messages | grep "$ERROR_MSG" >> error.log
  2. # “error.log”文件将不会写入任何东西。
  3. # 正如Samuli Kaipiainen指出的那样,
  4. #+ 这一结果是从grep的缓冲区输出的。
  5. # 解决的办法就是把“--line-buffered”参数添加到grep中。
  • 在脚本中使用“suid”命令是非常危险的,因为这会危及系统安全。^suid

  • 使用shell脚本来编写CGI程序是值得商榷的。因为Shell脚本的变量不是“类型安全”的,当CGI被关联的时候,可能会产生令人不快的行为。此外,它还很难抵挡住“破解的考验”。

  • Bash不能正确地处理双斜线(//)字符串

  • 在Linux或BSD上编写的Bash脚本,可能需要修改一下,才能使它们运行在商业的UNIX机器上。这些脚本通常都使用GNU命令和过滤工具,GNU工具通常都比一般的UNIX上的同类工具更加强大。这方面的一个非常明显的例子就是,文本处理工具tr

  • 遗憾的是,更新Bash本身就会破坏过去工作完全正常的脚本。让我们回顾一下使用无正式文件的Bash功能有多危险

危险正在接近你 —
小心,小心,小心,小心。
许多勇敢的心都在沉睡。
所以一定要小心 —
小心。

——A.J. Lamb and H.W. Petrie

注意事项