bash - 在命令替换中使用外部调用的陷阱会破坏父 Bash shell

标签 bash shell command bash-trap command-substitution

我有一个基于文本的用户界面脚本,允许我浏览目录和选择一个文件。图形输出到 stderr,所选文件的路径发送到 stdout。这允许以这种方式获取所选文件:

file="$(./script)"

这非常方便,因为命令替换只获取 stdout

但我需要我的脚本来处理信号,以便当脚本被中断时,它可以重置显示。我设置了一个处理 INT 信号的陷阱。要模拟它在做什么,请考虑以下脚本:

catch() { 
    echo "caught"
    ps # Calling an external command
    exit
}

trap catch INT

while read -sN1; do # Reading from the keyboard
    echo $REPLY >&2
done

然后使用 var="$(./script)" 调用脚本。现在,如果您通过点击 ^C 发送 INT 信号,父 shell 就会中断:您键入的任何内容(包括控制字符)都将被打印出来,直到您点击回车,然后您的任何输入都不会显示。

删除 catch 函数中的外部命令调用似乎可以解决问题(不过,echo 似乎不起作用),但我不明白为什么,我不能在我的最终脚本中没有它。

有什么我想念的吗?为什么这会破坏父 shell?

最佳答案

我未经证实但最好的理论是,这是由读取终端设置的父进程和恢复它们的子进程之间的竞争引起的。

当中断时,交互式 shell 将停止尝试从管道读取,并仔细检查当前的终端设置以避免以后破坏它们。如果 child 还没有恢复它们, parent 将阅读错误的设置并假设终端应该是这样的。

这就是为什么您可以在它开始困惑之前键入一行的原因:子项已将良好设置恢复为缓冲规范模式,因此您可以键入整行。一旦你按下回车键,bash 就会得到命令,并且作为它提示的一部分,恢复它认为终端应该有的错误设置。

为了解决这个问题,您可以让父级在捕获期间处理 SIGINT。处理程序做什么并不重要,因为唯一的一点是让 Bash 等待当前命令完成,以便它可以调用处理程序。

这是一个例子:

#!/bin/bash

catch() {
  sleep 1 # Make sure to lose the race
  echo "caught"
  ps
  exit
}

trap catch INT

while read -sN1; do # Reading from the keyboard
    echo $REPLY >&2
done

这是在输入 x 并按下 Ctrl-C 后的交互式 shell:

bash-5.0$ trap 'true' INT; var=$(./script)
x
bash-5.0$ echo "The prompt works fine"
The prompt works fine
bash-5.0$ declare -p var
declare -- var="caught
    PID TTY          TIME CMD
 650388 pts/3    00:00:00 bash
 650859 pts/3    00:00:00 script
 650862 pts/3    00:00:00 ps"
bash-5.0$

这里没有父级中的陷阱,展示了如何只有第一个命令直到第一次输入有效,而其余输入是隐藏的:

bash-5.0$ trap - INT; var=$(./script)
x

bash-5.0$ echo "I can see this first line"
I can see this first line
bash-5.0$ bash: fasdfasdfasdfasdfa: command not found

关于bash - 在命令替换中使用外部调用的陷阱会破坏父 Bash shell,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61234233/

相关文章:

linux - Bash shell 脚本: How to Sum the Values in a File With the Use of For Loop and bc Command

bash - Linux 命令在 shell 命令行中表现异常

javascript - 如何在 VS 命令窗口中列出 JavaScript 对象的方法?

linux - 如何在没有键盘的情况下在 bash 脚本中获取文件名

linux - sh\bash 脚本中的 PACKAGE 关键字到底意味着什么?

bash - Shell 脚本退出代码 - 无法设置

如果不是或使用两个命令进行 Bash

Linux shell 脚本 "read"命令

linux - `git remote add` 抛出 fatal error

linux - 使用 Bash 在 "sudo -i"之后切换回之前的用户