Bash:当脚本终止时,如何终止脚本的子进程?

时间:2011-10-19 07:00:48

标签: linux bash subprocess termination

该问题适用于以下脚本:

脚本

#!/bin/sh

SRC="/tmp/my-server-logs"

echo "STARTING GREP JOBS..."
for f in `find ${SRC} -name '*log*2011*' | sort --reverse`
do
    (
        OUT=`nice grep -ci -E "${1}" "${f}"`
        if [ "${OUT}" != "0" ]
        then
            printf '%7s : %s\n' "${OUT}" "${f}"
        else
            printf '%7s   %s\n' "(none)" "${f}"
        fi
    ) &
done

echo "WAITING..."
wait

echo "FINISHED!"

当前行为

在控制台中按Ctrl+C会终止脚本,但不会终止已经运行的grep进程。

2 个答案:

答案 0 :(得分:16)

Ctrl+c写一个陷阱,并在陷阱中杀死所有子进程。将它放在wait命令之前。

function handle_sigint()
{
    for proc in `jobs -p`
    do
        kill $proc
    done
}

trap handle_sigint SIGINT

答案 1 :(得分:0)

一个简单的替代方法是使用cat管道。以下对我有用:

echo "-" > test.text; 
for x in 1 2 3; do 
    ( sleep $x; echo $x | tee --append test.text; ) & 
done | cat

如果我在最后一个数字打印到标准输出之前按Ctrl-C。如果文本生成命令需要很长时间,例如“find /”,它也可以工作,即它不仅通过cat连接到stdout,而且实际上是子进程。

对于广泛使用子进程的大型脚本,确保缩进的Ctrl-C行为的最简单方法是将整个脚本包装到这样的子shell中,例如。

#!/usr/bin/bash
(
    ...
) | cat

我不确定它是否与Andrew的答案具有完全相同的效果(即我不确定向子进程发送了什么信号)。此外,我只使用cygwin进行了测试,而不是使用原生Linux shell。