8

语境:

用户向我提供他们要运行的自定义脚本。这些脚本可以是任何类型的脚本,例如启动多个 GUI 程序、后端服务的脚本。我无法控制脚本的编写方式。这些脚本可以是阻塞类型,即执行等待直到所有子进程(顺序运行的程序)退出

#exaple of blocking script
echo "START"
first_program 
second_program 
echo "DONE"

或非阻塞类型,即在后台分叉子进程并退出类似

#example of non-blocking script
echo "START"
first_program &
second_program &
echo "DONE"

我想达到什么目的?

用户提供的脚本可以是上述两种类型中的任何一种,也可以是两者的混合。我的工作是运行脚本并等待它启动的所有进程退出,然后关闭节点。如果它是阻塞类型的,情况很简单,即获取脚本执行进程的 PID 并等待 ps -ef|grep -ef PID 没有更多条目。非阻塞脚本是给我带来麻烦的脚本

有没有一种方法可以获取通过执行脚本生成的所有子进程的 PID 列表?任何指针或提示将不胜感激

4

5 回答 5

7

您可以使用wait等待启动的所有后台进程userscript完成。由于wait仅适用于当前 shell 的子进程,因此您需要获取他们的脚本,而不是将其作为单独的进程运行。

( source userscript; wait )

在显式子外壳中获取脚本应该足够接近地模拟启动新进程。如果没有,您也可以将 subshel​​l 置于后台,这会强制启动一个新进程,然后等待完成。

( source userscript; wait ) & wait
于 2013-09-06T18:08:27.603 回答
7

ps --ppid $PID将列出该进程的所有子进程$PID

于 2013-09-06T17:27:40.720 回答
5

您可以打开由其他进程继承的文件描述符,然后等到它不再使用。这是一种低开销的方法,通常可以正常工作,尽管进程可以根据需要绕过它:

 foo=$(mktemp)
 ( flock -x 5000; theirscript; ) 5000> "$foo"
 flock -x 0 < "$foo"
 rm "$foo"
 echo "The script and its subprocesses are done"

您可以使用 ptrace 跟踪所有调用的进程,例如 with strace。这更容易,但有一些相关的开销,并且在脚本调用 suid 二进制文件时可能不起作用:

strace -f -e none theirscript
于 2013-09-06T17:42:00.780 回答
1

您可以使用pgrep -P <parent_pid>来获取子进程的列表。例子:

IFS=$'\n' read -ra CHILD_PROCS -d '' < <(exec pgrep -P "$1")

要获得孙子,只需对每个子进程执行相同的程序。

查看我的博客Bash 函数以列出和终止或发送信号以处理树

您可以使用其中一项功能正确列出在一个进程下生成的所有进程。每个都有自己的方法或发送信号的顺序来处理。

唯一的限制是进程仍然必须连接而不是孤立的。如果您能以某种方式找到对流程进行分组的方法,那么这可能是您的解决方案。

于 2013-09-06T17:52:31.867 回答
1

简单地回答被问到的问题。您可以将调用的每个脚本的进程 ID 存储到同一个变量中:

echo "START"
first_program &
child_process_ids+="$! "
second_program &
child_process_ids+="$! "
echo $child_process_ids
echo "DONE"

$child_process_ids 只是一个以空格分隔的进程 ID 字符串。现在,这回答了所提出的问题,但是,我会做的事情会有所不同。我会从 for 循环中调用每个脚本,存储其进程 ID,然后在另一个 for 循环中等待每个脚本完成并分别检查每个退出代码。使用相同的示例,这就是它的样子。

echo "START"

scripts="first_program second_program"

for script in $scripts; do
    #Call script and send to background
    ./$script &

    #Store the script's processID that was just sent to the background
    child_process_ids+="$! "
done

for child_process_id in $child_process_ids; do

    #Pass each processId into the wait command to retrieve its exit 
    #code and store it in $rc
    wait $child_process_id
    rc=$?

    #Inspect each processes exit code
    if [ $rc -ne 0 ]; then
        echo "$child_process_id failed with an exit code of $rc"
    else
        echo "$child_process_id was successful"
    fi
done
于 2016-11-07T22:15:57.233 回答