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 / 问题 / 778251
Accepted
intelfx
intelfx
Asked: 2024-06-13 09:09:56 +0800 CST2024-06-13 09:09:56 +0800 CST 2024-06-13 09:09:56 +0800 CST

有没有办法在启动管道后“执行”管道或退出 shell?

  • 772

我正在编写一个 shell 包装脚本,它应该充当寻呼机(在 stdin 上接收输入并在 tty 上执行输出)。

该包装器准备环境和命令行,然后在管道中启动两个进程:第一个是非交互式过滤器,最后一个是需要控制 tty 的实际寻呼机(或者,它生成实际的寻呼机本身):

#!/bin/bash

...

col -bx | bat "${bat_args[@]}" --paging=always

最终的进程树如下:

<parent>
\- wrapper.sh
   |- col -bx (dies)
   \- bat ...
      \- less ...

该col -bx进程在过滤输入后退出,并由 shell 收获。


是否有可能摆脱 shell 进程,以使其在寻呼机运行时不会一直处于挂起状态?

我想到了一个使用进程替换的解决方法:

exec bat "${bat_args[@]}" --paging=always < <(col -bx)

但是,col -bx进程不会被 shell 回收,而是保持 Z 状态。有没有“正确”的方式来编写这个包装器?

bash
  • 1 1 个回答
  • 44 Views

1 个回答

  • Voted
  1. Best Answer
    Stéphane Chazelas
    2024-06-13T13:30:55+08:002024-06-13T13:30:55+08:00

    您还可以执行以下操作:

    #!/bin/zsh -
    col -bx | exec bat "${bat_args[@]}" --paging=always
    

    或者:

    #!/bin/ksh -
    col -bx | exec bat "${bat_args[@]}" --paging=always
    

    (不适用于 pdksh 派生的 ksh 实现)

    #!/bin/bash -
    
    shopt -s lastpipe
    col -bx | exec bat "${bat_args[@]}" --paging=always
    

    最后,功能上等同于您使用进程替换进行重定向所做的事情,但跳过了额外的 fd 改组和命名管道创建(或在/dev/fd/x具有它们的系统上打开(大多数))。

    无论如何,要收割一个​​进程,您需要等待它。使用ksh/ zsh/bash -O lastpipe和上述方法,col将成为最终执行的进程的子进程bat,因此在死亡bat时将获得 SIGCHLD col,它如何处理它取决于它。如果它不处理该 SIGCHLD(很少有应用程序会处理它们自己没有生成的进程),col它将显示为僵尸进程直到bat终止,此后该col进程将重新成为子子收割者的父进程或init并在那里进行处理。

    无论如何,请注意,除非您设置选项,否则A | Bksh 不会等待。Apipefail

    现在,根据 XSI 选项下的 POSIX,这也适用于 Linux,

    如果调用进程的父进程已设置其 SA_NOCLDWAIT 标志或已将 SIGCHLD 信号的操作设置为 SIG_IGN:

    • 该进程的状态信息(参见状态信息)(如果有)应被丢弃。
    • 调用进程的生命周期应立即结束。如果设置了 SA_NOCLDWAIT,则是否向父进程发送 SIGCHLD 信号由实现定义。
    • 如果调用进程的父进程中的线程在 wait()、waitpid() 或 waitid() 中被阻塞,并且父进程在等待的子进程集合中没有剩余的子进程,则 wait()、waitid() 或 waitpid() 函数将失败并将 errno 设置为 [ECHILD]。

    否则:

    • 应生成状态信息(参见状态信息)。
    • 调用进程将被转换为僵尸进程。其状态信息将提供给父进程,直到进程的生命周期结束。
      [...]

    (重点是我的)。

    现在,并不是所有的 shell 都允许您忽略 SIGCHLD,因为它对于其功能至关重要(shell 的全部目的是运行命令并报告其退出状态),但看起来当前版本bash(我测试的是 5.2.21)至少可以这样做,因此:

    #!/bin/bash -
    
    shopt -s lastpipe
    trap '' CHLD
    col -bx | exec bat "${bat_args[@]}" --paging=always
    

    col只要在退出bat之前不取消忽略 SIGCHLD,正在运行的进程就永远不会变成僵尸。col

    无论如何,僵尸不一定是坏事。僵尸阶段是进程生命的一部分,大多数进程在死亡和其父进程(或init子进程)确认死亡之间都会经历这个阶段。

    • 1

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

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

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

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

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

Sidebar

Stats

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

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

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

    • 4 个回答
  • Marko Smith

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

    • 5 个回答
  • Marko Smith

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

    • 3 个回答
  • 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
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +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

热门标签

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