- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有以下脚本监控 /tmp
无限期目录,如果对该目录中的文件有任何操作,则文件名由 while 循环读取,并首先 a
文件名中的字符替换为 b
字符和此修改后的文件名记录到 test.log
文件:
#!/bin/bash
trap ':' HUP
trap 'kill $(jobs -p)' EXIT
/usr/local/bin/inotifywait -q -m /tmp --format %f |
while IFS= read -r filename; do
echo "$filename" | sed 's/a/b/' > test.log
done
这是实际脚本的简化版本。我还有一个 Sys-V 类型的初始化脚本用于上面的脚本,因为我想保持 LSB 兼容,我的初始化脚本有 force-reload
(如果服务支持,则重新加载配置。否则,重新启动服务。)向脚本发送 HUP 信号的选项。现在在执行 force-reload
之前,它执行 killproc -HUP test.sh
, pstree
的输出如下:
# pstree -Ap 4424
test.sh(4424)-+-inotifywait(4425)
`-test.sh(4426)
#
执行 strace killproc -HUP test.sh
后子 shell 终止:
# pstree -Ap 4424
test.sh(4424)---inotifywait(4425)
#
根据 strace
, killproc
已发送SIGHUP
到流程 4424
和 4426
, 但只有后者被终止。
这个带有 PID 4426
的子 shell 有什么意义?在我的示例中,即为什么首先创建它?另外,有没有办法忽略HUP
信号?
最佳答案
问题的第一部分由 shell(在本例中为 Bash)在管道中运行命令的机制来解释。
管道是一个 FIFO(先进先出)单向进程间通信 (IPC) channel :它允许在一端写入字节(只写端)并从另一个(只读端)读取,而无需读取或写入物理文件系统。
管道 允许两个不同的命令通过匿名 或未命名(即,在文件系统中没有条目)相互通信管道。
当shell执行一个简单的命令时,该命令在shell的子进程中运行。如果不使用作业控制,则当子进程终止时,shell 将重新获得对终端的控制。
当两个命令在管道中运行时,管道中的两个命令将作为两个同时运行的独立子进程执行。
在 Unix 系统中,管道是使用 pipe(2)
系统调用创建的,它会创建一个新管道并返回一对文件描述符,其中一个指向 read 端,另一个到管道的写入端。
对于 GNU/Linux 系统上的 Bash,clone(2)
系统调用用于创建子进程。这允许子进程与其父进程共享文件描述符表,以便两个子进程都继承匿名管道的文件描述符,以便一个可以读取,另一个可以写入
在您的情况下,inotifywait
命令获取 PID 4425 并通过将其 stdout
连接到管道的文件描述符来写入管道的只写端写完。
同时,管道命令的右侧获取 PID,4426 并将其 stdin
文件描述符设置为管道只读端的文件描述符。由于管道右侧的子 shell 不是外部命令,因此代表子进程的名称与其父进程相同,test.sh
。
有关详细信息,请参阅 man 7 pipe
和以下链接:
我花了很长时间(实际上是几个小时的研究)才弄清楚为什么 SIGHUP 信号的陷阱没有被忽略。
我的所有研究表明,由 clone(2)
系统调用创建的子进程也应该能够共享父进程的信号处理程序表。
Bash 手册页还指出
Command substitution, commands grouped with parentheses, and asynchronous commands are invoked in a subshell environment that is a duplicate of the shell environment, except that traps caught by the shell are reset to the values that the shell inherited from its parent at invocation.
后来又说
Signals ignored upon entry to the shell cannot be trapped or reset. Trapped signals that are not being ignored are reset to their original values in a subshell or subshell environment when one is created.
这表明子 shell 不会继承未被忽略的信号处理程序。据我了解,您的 trap ':' HUP
行意味着 SIGHUP 信号被(有效地)忽略了(因为 :
内置函数除了返回成功外什么都不做)——并且反过来应该被管道的子 shell 忽略。
但是,我最终在 Bash 手册页中看到了关于 trap
的描述,它定义了 Bash 中 ignore 的含义:
If arg is the null string the signal specified by each sigspec is ignored by the shell and by the commands it invokes.
只需将 trap
命令更改为 trap '' HUP
即可确保忽略 SIGHUP 信号,对于脚本本身以及任何子 shell。
关于bash - 使用管道命令忽略 Bash 脚本中的 HUP 信号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34162503/
我正在使用 Assets 管道来管理我的 Grails 3.0 应用程序的前端资源。但是,似乎没有创建 CoffeeScript 文件的源映射。有什么办法可以启用它吗? 我的 build.gradle
我有一个我想要的管道: 提供一些资源, 运行一些测试, 拆资源。 我希望第 3 步中的拆卸任务运行 不管 测试是否通过或失败,在第 2 步。据我所知 runAfter如果前一个任务成功,则只运行一个任
如果我运行以下命令: Measure-Command -Expression {gci -Path C:\ -Recurse -ea SilentlyContinue | where Extensio
我知道管道是一个特殊字符,我需要使用: Scanner input = new Scanner(System.in); String line = input.next
我再次遇到同样的问题,我有我的默认处理方式,但它一直困扰着我。 有没有更好的办法? 所以基本上我有一个运行的管道,在管道内做一些事情,并想从管道内返回一个键/值对。 我希望整个管道返回一个类型为 ps
我有三个环境:dev、hml 和 qa。 在我的管道中,根据分支,阶段有一个条件来检查它是否会运行: - stage: Project_Deploy_DEV condition: eq(varia
我有 Jenkins Jenkins ver. 2.82 正在运行并想在创建新作业时使用 Pipeline 功能。但我没有看到这个列为选项。我只能在自由式项目、maven 项目、外部项目和多配置之间进
在对上一个问题 (haskell-data-hashset-from-unordered-container-performance-for-large-sets) 进行一些观察时,我偶然发现了一个奇
我正在寻找有关如何使用管道将标准输出作为其他命令的参数传递的见解。 例如,考虑这种情况: ls | grep Hello grep 的结构遵循以下模式:grep SearchTerm PathOfFi
有没有办法不因声明性管道步骤而失败,而是显示警告?目前我正在通过添加 || exit 0 来规避它到 sh 命令行的末尾,所以它总是可以正常退出。 当前示例: sh 'vendor/bin/phpcs
我们正在从旧的 Jenkins 设置迁移到所有计划都是声明性 jenkinsfile 管道的新服务器……但是,通过使用管道,我们无法再手动清除工作区。我如何设置 Jenkins 以允许 手动点播清理工
我在 Python 中阅读了有关 Pipelines 和 GridSearchCV 的以下示例: http://www.davidsbatista.net/blog/2017/04/01/docume
我有一个这样的管道脚本: node('linux'){ stage('Setup'){ echo "Build Stage" } stage('Build'){ echo
我正在使用 bitbucket 管道进行培训 这是我的 bitbucket-pipelines.yml: image: php:7.2.9 pipelines: default:
我正在编写一个程序,其中输入文件被拆分为多个文件(Shamir 的 secret 共享方案)。 这是我想象的管道: 来源:使用 Conduit.Binary.sourceFile 从输入中读取 导管:
我创建了一个管道,它有一个应该只在开发分支上执行的阶段。该阶段还需要用户输入。即使我在不同的分支上,为什么它会卡在这些步骤的用户输入上?当我提供输入时,它们会被正确跳过。 stage('Deplo
我正在尝试学习管道功能(%>%)。 当试图从这行代码转换到另一行时,它不起作用。 ---- R代码--原版----- set.seed(1014) replicate(6,sample(1:8))
在 Jenkins Pipeline 中,如何将工件从以前的构建复制到当前构建? 即使之前的构建失败,我也想这样做。 最佳答案 Stuart Rowe 还在 Pipeline Authoring Si
我正在尝试使用 执行已定义的作业构建 使用 Jenkins 管道的方法。 这是一个简单的例子: build('jenkins-test-project-build', param1 : 'some-
当我使用 where 过滤器通过管道命令排除对象时,它没有给我正确的输出。 PS C:\Users\Administrator> $proall = Get-ADComputer -filter *
我是一名优秀的程序员,十分优秀!