AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 483744
Accepted
kjo
kjo
Asked: 2018-11-24 10:37:55 +0800 CST2018-11-24 10:37:55 +0800 CST 2018-11-24 10:37:55 +0800 CST

如何编写一个可靠地退出(以指定状态)当前进程的函数?

  • 772

下面的脚本是该问题的最小(尽管是人为的)说明。

## demo.sh

exitfn () {
    printf -- 'exitfn PID: %d\n' "$$" >&2
    exit 1
}

printf -- 'script PID: %d\n' "$$" >&2

exitfn | :

printf -- 'SHOULD NEVER SEE THIS (0)\n' >&2

exitfn

printf -- 'SHOULD NEVER SEE THIS (1)\n' >&2

在此示例脚本中,exitfn代表一个函数,其工作需要终止当前进程1。

不幸的是,作为实施,exitfn并不能可靠地完成这个任务。

如果运行此脚本,输出如下所示:

% bash ./demo.sh
script PID: 26731
exitfn PID: 26731
SHOULD NEVER SEE THIS (0)
exitfn PID: 26731

(当然,每次调用显示的 PID 值都会不同。)

这里的关键点是,在第一次调用该exitfn函数时,exit 1其主体中的命令无法终止封闭脚本printf的执行(紧随其后的第一个命令的执行证明了这一点)。相反,在第二次调用 时exitfn,此命令确实使脚本的执行结束(如第二个命令未执行exit 1的事实所证明的)。printf

两次调用之间的唯一区别exitfn是第一个调用作为双组件管道的第一个组件出现,而第二个调用是“独立”调用。

我对此感到困惑。我曾预计这exit会导致终止当前进程(即具有由 给出的 PID 的进程$$)。显然,这并不总是正确的。

尽管如此,有没有一种方法可以exitfn让它退出周围的脚本,即使在管道中调用时也是如此?


顺便说一句,上面的脚本也是一个有效的 zsh 脚本,并产生相同的结果:

% zsh ./demo.sh
script PID: 26799
exitfn PID: 26799
SHOULD NEVER SEE THIS (0)
exitfn PID: 26799

我也对 zsh 的这个问题的答案感兴趣。


最后,我应该指出,exitfn像这样实现根本行不通:

exitfn () {
    printf -- 'exitfn PID: %d\n' "$$" >&2
    exit 1
    kill -9 "$$"
}

...因为在任何情况下,该exit 1命令始终是该函数执行的最后一行。(替换 exit 1为kill -9 $$是不可接受的:我想控制脚本的退出状态,以及它到 stderr 的输出。)


1实际上,这样的功能会在终止当前进程之前执行其他任务,例如诊断日志记录或清理操作。

bash shell-script
  • 1 1 个回答
  • 76 Views

1 个回答

  • Voted
  1. Best Answer
    Gilles 'SO- stop being evil'
    2018-11-24T14:13:09+08:002018-11-24T14:13:09+08:00

    我曾预计这exit会导致终止当前进程(即 PID 由 给出的进程$$)

    “当前进程”与“PID由 给出的进程”不是一回事$$。exit退出当前subshel​​l或原始 shell(如果未在子 shell 中调用)。

    某些构造,例如( … )(使用子 shell 分组)的内容、命令替换($(…)或`…`)和管道的每一侧(或仅在某些 shell 中为左侧),在子 shell 中运行。子 shell 的行为就好像它是使用创建的单独进程fork(),并且通常以这种方式实现(某些 shell 在某些情况下不使用子进程,作为性能优化)。子shell有自己的变量副本,自己的重定向等,调用exit退出子shell,就像exit()标准C库中的函数退出进程一样。有关子外壳的更多信息,请参阅什么是子外壳(在 make 文档的上下文中)?,“子外壳”和“子进程”之间的确切区别是什么?$() 是一个子外壳吗?.

    $$始终是原始 shell 进程的进程 ID。它在子外壳中不会改变。一些 shell 有一个在子 shell 中变化的变量,例如$BASHPID在 bash 和 mksh 中,或者${.sh.subshell}在 ksh 或$ZSH_SUBSHELLzsh¹$sysparams[pid]中。

    有没有一种编写方法,exitfn即使在管道中调用它也能退出周围的脚本?

    不完全是。您将需要做更多的工作,或者在脚本创建子 shell 的地方,或者在顶层,或者两者兼而有之。有关近似值,请参阅从子 shell 退出 shell 脚本。


    ¹ 请注意,尽管与其他 shell 相反,zsh它在父进程中的管道中执行扩展(从左到右),而不是在管道的每个成员中执行扩展:zsh -c 'echo $ZSH_SUBSHELL | cat'输出0(即使echo在子进程中运行)和zsh -c 'n=0; echo $((++n)) | echo $((++n))'输出 2。

    • 3

相关问题

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • MySQL Select with function IN () with bash array

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve