如何在Bash中等待多个子进程完成,并且当其中任何一个子进程以非零退出状态结束时,使主进程也返回一个非零的退出码?

2024-05-27 15:57:07 浏览数 (2)

问题

如何在 Bash 脚本中等待该脚本启动的多个子进程完成,并且当这其中任意一个子进程以非零退出码结束时,让该脚本也返回一个非零的退出码?

简单的脚本:

代码语言:javascript复制
#!/bin/bash
for i in `seq 0 9`; do
  calculations $i &
done
wait

上述脚本将会等待所有 10 个被创建的子进程结束,但它总会给出退出状态 0(参见 wait 的帮助信息)。我应该如何修改这个脚本,使其能检测到被创建子进程的退出状态,并且当任何子进程以非零代码结束时,让脚本返回退出码 1?


回答

根据 Luca Tettamanti 和 Gabriel Staples 的回答,编写一个完整的可以运行的演示代码:

代码语言:javascript复制
#!/usr/bin/env bash

# 这是一个特殊的 sleep 函数,它将睡眠的秒数作为"错误代码"
# 或"返回代码"返回,以便我们可以清楚地看到,实际上
# 我们在每个进程完成时确实获取了它的返回代码。
my_sleep() {
    seconds_to_sleep="$1"
    sleep "$seconds_to_sleep"
    return "$seconds_to_sleep"
}

# 创建一个你想作为子进程运行的命令数组
procs=()  # bash数组
procs =("my_sleep 2")
procs =("my_sleep 1")
procs =("my_sleep 4")
procs =("my_sleep 3")

num_procs=${#procs[@]}  # 数组中元素的个数
echo "num_procs = $num_procs"

# 作为子进程运行命令并把 pid 存储到数组中
pids=()  # bash数组
for (( i=0; i<"$num_procs"; i   )); do
    echo "cmd${i} : ${procs[$i]}"
    ${procs[$i]} &  # 将 cmd 作为子进程运行
    pids =("$!")    # 存储上一个子进程启动的 pid
    echo "    pid = ${pids[$i]}"
done

for pid in $pids; do
  wait $pid
  rc=$?
  [ $rc -ne 0 ] && break  # 若子进程以非零退出码结束,则跳出循环
done

#echo $rc
exit $rc

将代码保存为文件 wait_procs_demo.sh,再运行测试:


参考

  • stackoverflow question 356100
  • help wait
  • https://www.gnu.org/software/bash/manual/bash.html#Lists

0 人点赞