36.4 递归:可以调用自己的脚本

脚本可以递归的调用自己吗?答案是肯定的。

Example 36-10. 可以调用自己的脚本(但没什么实际用途)

  1. #!/bin/bash
  2. # recurse.sh
  3. # 脚本可以调用自己吗?
  4. # 其实是可以的。但这样有什么实际用途吗?
  5. # (请往下看)
  6. RANGE=10
  7. MAXVAL=9
  8. i=$RANDOM
  9. let "i %= $RANGE" # 在0到$RANGE - 1的范围内产生一个随机数
  10. if [ "$i" -lt "$MAXVAL" ]
  11. then
  12. echo "i = $i"
  13. ./$0 # 脚本进行递归调用(调用自己)
  14. fi # 每次被调用的脚本做同样的事情,直到$i和$MAXVAL相等。
  15. # 如果使用“while”循环代替“if/then”语句会出问题。请试着解释为什么。
  16. exit 0
  17. # 笔记:
  18. # ----
  19. # 这个脚本文件必须有可执行权限。
  20. # 即使使用“sh”命令调用,这脚本也可以执行。
  21. # 请解释原因。

Example 36-11. 一个有点用的调用自己的脚本

  1. #!/bin/bash
  2. # pb.sh: phone book
  3. # 用于权限管理的脚本,由Rick Boivie编写。
  4. # ABS作者稍有修改
  5. MINARGS=1 # 需要至少一个参数
  6. DATAFILE=./phonebook
  7. # 当前目录下必须存在名为“phonebook”的数据文件
  8. PROGNAME=$0
  9. E_NOARGS=70 # 没有错误
  10. if [ $# -lt $MINARGS ]; then
  11. echo "Usage: "$PROGNAME" data-to-look-up"
  12. exit $E_NOARGS
  13. fi
  14. if [ $# -eq $MINARGS ]; then
  15. grep $1 "$DATAFILE"
  16. # 如果$DATAFILE没有匹配则'grep'命令会报错。
  17. else
  18. ( shift; "$PROGNAME" $* ) | grep $1
  19. # 脚本的递归调用
  20. fi
  21. exit 0 # 脚本结束
  22. # 下面是一些文件内容
  23. # ------------------------------------------------------------------------
  24. 一个简单的"phonebook"数据文件:
  25. John Doe 1555 Main St., Baltimore, MD 21228 (410) 222-3333
  26. Mary Moe 9899 Jones Blvd., Warren, NH 03787 (603) 898-3232
  27. Richard Roe 856 E. 7th St., New York, NY 10009 (212) 333-4567
  28. Sam Roe 956 E. 8th St., New York, NY 10009 (212) 444-5678
  29. Zoe Zenobia 4481 N. Baker St., San Francisco, SF 94338 (415) 501-1631
  30. # ------------------------------------------------------------------------
  31. $bash pb.sh Roe
  32. Richard Roe 856 E. 7th St., New York, NY 10009 (212) 333-4567
  33. Sam Roe 956 E. 8th St., New York, NY 10009 (212) 444-5678
  34. $bash pb.sh Roe Sam
  35. Sam Roe 956 E. 8th St., New York, NY 10009 (212) 444-5678
  36. # 当对脚本传入多于一个参数时,脚本只显示包含所有参数的行

Example 36-12. 另一个调用自己的脚本

  1. #!/bin/bash
  2. # usrmnt.sh, 由Anthony Richardson编写
  3. # 在ABS Guide中用于权限管理
  4. # usage: usrmnt.sh
  5. # description: 想使用挂载设备操作的用户,在/etc/sudoers文件中必须属于MNTUSERS组。
  6. # ----------------------------------------------------------
  7. # 这个脚本会返回加了sudo命令的自身。
  8. # 如果一个有权限的用户,则只需要输入:
  9. # usermount /dev/fd0 /mnt/floppy
  10. # 而不需要使用下面的方法:
  11. # sudo usermount /dev/fd0 /mnt/floppy
  12. # 我对于所有需要sudo的脚本都使用了这个技术,因为我发现这让我感觉非常舒服。
  13. # ----------------------------------------------------------
  14. # 如果SUDO_COMMAND变量没有被设置,那证明没有使用sudo命令运行。这需要
  15. # 再重新运行这个脚本,同时传递用户的ID和组ID...
  16. if [ -z "$SUDO_COMMAND" ]
  17. then
  18. mntusr=$(id -u) grpusr=$(id -g) sudo $0 $* # 译注:脚本调用自己,并且传递参数
  19. exit 0
  20. fi
  21. # 如果使用了sudo运行,就不会卡在这里了。
  22. /bin/mount $* -o uid=$mntusr,gid=$grpusr
  23. exit 0
  24. # 附加说明:
  25. # -------------------------------------------------
  26. # 1) Linux系统允许/etc/fstab文件中列出的用户挂在移动存储设备。但在服务器上,我喜欢让更少的人访问移动存储。我发现使用sudo可以帮我做到。
  27. # 2) 我还发现使用sudo比用组权限来实现让人感觉更加舒服。
  28. # 3) 这种方法可以给任何有权限的人使用mount命令,所以要小心处理。
  29. # 你也可以将这种技术用到比如mntfloppy,mntcdrom和mntsamba等脚本上来实现更优雅的挂载管理。

过多层次的递归调用会导致脚本的栈空间溢出,引起段错误(segfault)。