第六章 退出与退出状态

Bourne shell里存在不明确之处,但人们也会使用它们。

—— Chat Ramey

跟C程序类似,exit 命令被用来结束脚本。同时,它也会返回一个值,返回值可以被交给父进程。

每个命令都会返回一个退出状态(exit status),有时也叫做返回状态(return status)或退出码(exit code)。命令执行成功返回0,如果返回一个非0值,通常情况下会被认为是一个错误代码。一个运行状态良好的UNIX命令、程序和工具在正常执行退出后都会返回一个0的退出码,当然也有例外。

同样地,脚本中的函数和脚本本身也会返回一个退出状态。在脚本或者脚本函数中执行的最后的命令会决定它们的退出状态。在脚本中,exit nnn 命令将会把nnn退出状态码传递给shell(nnn 必须是 0-255 之间的整型数)。

note 当一个脚本以不带参数的 exit 来结束时,脚本的退出状态由脚本最后执行命令决定(exit 命令之前)。

  1. #!/bin/bash
  2. COMMAND_1
  3. ...
  4. COMMAND_LAST
  5. # 将以最后的命令来决定退出状态
  6. exit

exitexit $? 以及省略 exit 效果等同。

  1. #!/bin/bash
  2. COMMAND_1
  3. ...
  4. COMMAND_LAST
  5. #将以最后的命令来决定退出状态
  6. exit $?
  1. #!/bin/bash
  2. COMMAND_1
  3. ...
  4. COMMAND_LAST
  5. #将以最后的命令来决定退出状态

$? 读取上一个执行命令的退出状态。在一个函数返回后,$? 给出函数最后执行的那条命令的退出状态。这就是Bash函数的”返回值”。^1

管道执行后,$? 给出最后执行的那条命令的退出状态。

在脚本终止后,命令行下键入$?会给出脚本的退出状态,即在脚本中最后一条命令执行后的退出状态。一般情况下,0为成功,1-255为失败。

样例 6-1. 退出与退出状态

  1. #!/bin/bash
  2. echo hello
  3. echo $? # 返回值为0,因为执行成功。
  4. lskdf # 不认识的命令。
  5. echo $? # 返回非0值,因为失败了。
  6. echo
  7. exit 113 # 将返回113给shell
  8. # 为了验证这些,在脚本结束的地方使用“echo $?”
  9. # 按照惯例,'exit 0' 意味着执行成功,
  10. #+ 非0意味着错误或者异常情况。
  11. # 查看附录章节“退出码的特殊含义”

$? 对于测试脚本中的命令的执行结果特别有用(查看样例 16-35和样例 16-20)。

note 逻辑非操作符 ! 将会反转测试或命令的结果,并且这将会影响退出状态。

样例 6-2. 否定一个条件使用!

  1. true # true 是 shell 内建命令。
  2. echo "exit status of \"true\" = $?" # 0
  3. ! true
  4. echo "exit status of \"! true\" = $?" # 1
  5. # 注意在命令之间的 "!" 需要一个空格。
  6. # !true 将导致一个"command not found"错误。
  7. #
  8. # 如果一个命令以'!'开头,那么将调用 Bash 的历史机制,显示这个命令被使用的历史。
  9. true
  10. !true
  11. # 这次就没有错误了,但是同样也没有反转。
  12. # 它不过是重复之前的命令(true)。
  13. # ============================================================ #
  14. # 在 _pipe_ 前使用 ! 将改变返回的退出状态。
  15. ls | bogus_command #bash: bogus_command: command not found
  16. echo $? #127
  17. >
  18. ! ls | bogus_command #bash: bogus_command:command not found
  19. echo $? #0
  20. # 注意 ! 不会改变管道的执行。
  21. # 只改变退出状态。
  22. #============================================================ #
  23. >
  24. # 感谢 Stéphane Chazelas 和 Kristopher Newsome。

caution 某些特定的退出码具有一些特定的保留含义,用户不应该在自己的脚本中重新定义它们。