• Bash Shell中的信号

    当没有任何捕获时,一个交互式 Bash Shell 会忽略 SIGTERM 和 SIGQUIT 信号。由 Bash 运行的非内部命令会使用 Shell 从其父进程继承的信号处理程序。如果没有启用作业控制,异步执行的命令会忽略除了有这些信号处理程序之外的 SIGINT 和 SIGQUIT 信号。由于命令替换而运行的命令会忽略键盘产生的作业控制信号SIGTTIN、SIGTTOU 和 SIGTSTP。

    默认情况下,Shell 接收到 SIGHUP 信号后会退出。在退出之前,一个交互式的 Shell 会向所有的作业,不管是正在运行的还是已停止的,重新发送 SIGHUP 信号。对已停止的作业,Shell 还会发送 SIGCONT 信号以确保它能够接收到 SIGHUP 信号。

    若要阻止 Shell 向某个特定的作业发送 SIGHUP 信号,可以使用内部命令 disown 将它从作业表中移除,或是用“disown -h”命令阻止 Shell 向特定的作业发送 SIGHUP 信号,但并不会将特定的作业从作业表中移除。

    我们通过如下实例,来了解一下 disown 命令的作用:

    #将 sleep 命令放在后台执行,休眠30秒
    [c.biancheng.net]$ sleep 30 &
    [1] 8052
    
    #列出当前 Shell 下所有作业的信息
    [c.biancheng.net]$ jobs -l
    [1]+ 8052 Running    sleep 30 &
    
    #将作业1从作业表中移除
    [c.biancheng.net]$ disown %1
    
    #再次列出当前 Shell 下所有作业的信息
    [c.biancheng.net]$ jobs -l
    
    #查找 sleep 进程
    [c.biancheng.net]$ ps -ef | grep sleep
    mozhiyan 8052 8092 cons1 11:28:21 /usr/bin/sleep
    
    #打印当前 Shell 的进程号
    [c.biancheng.net]$ echo $$
    8092

    在上述实例中,我们首先将命令“sleep 30”放在后台运行,此时,我们使用命令“jobs -l”可以看到作业表中有一个正在运行的作业,然后,我们使用命令“disown %1”将作业1从作业表中移除,再使用命令“jobs -l”会看到作业表中已经没有了作业,但是我们发现其实“sleep 30”这个命令的进程仍然存在。此时,Shell 若接收到 SIGHUP 信号,它就不会向作业1重新发送 SIGHUP 信号,此时如果我们退出 Shell,这个作业仍将继续运行,而不会被终止。

    我们再来看一下命令“disown -h”的用途:

    #将 sleep 命令放在后台执行,休眠30秒
    [c.biancheng.net]$ sleep 30 &
    [1] 3184
    
    #列出当前 Shell 下所有作业的信息
    [c.biancheng.net]$ jobs -l
    [1]+ 3184 Running    sleep 30 &
    
    #阻止 Shell 向作业1发送 SIGHUP 信号
    [c.biancheng.net]$ disown -h %1
    [c.biancheng.net]$ jobs -l
    [1]+ 3184 Running    sleep 30 &

    我们看到,在执行了命令“disown -h %1”后,作业1并没有从作业表中移除,但它己经被标记,所以即使 Shell 收到 SIGHUP 信号也不会向此作业发送 SIGHUP 信号。因此, 如果此时我们退出 Shell,这个作业也仍将继续运行,而不会被终止。

    注意:如果使用内部命令 shopt 打开了 Shell 的 huponexit 选项,当一个交互式的登录 Shell 退出时,会向所有的作业发送 SIGHUP 信号。

更多...

加载中...