文章目录
- 问题
- 回答
- 参考
问题
如何在 Bash 脚本中等待该脚本启动的多个子进程完成,并且当这其中任意一个子进程以非零退出码结束时,让该脚本也返回一个非零的退出码?
简单的脚本:
#!/bin/bash
for i in `seq 0 9`; docalculations $i &
done
wait
上述脚本将会等待所有 10 个被创建的子进程结束,但它总会给出退出状态 0
(参见 wait
的帮助信息)。我应该如何修改这个脚本,使其能检测到被创建子进程的退出状态,并且当任何子进程以非零代码结束时,让脚本返回退出码 1
?
回答
根据 Luca Tettamanti 和 Gabriel Staples 的回答,编写一个完整的可以运行的演示代码:
#!/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++ )); doecho "cmd${i} : ${procs[$i]}"${procs[$i]} & # 将 cmd 作为子进程运行pids+=("$!") # 存储上一个子进程启动的 pidecho " pid = ${pids[$i]}"
donefor pid in $pids; dowait $pidrc=$?[ $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
相关阅读:
- 如何使用bash脚本并行运行多个程序
- 如何用Shell命令结合 正则表达式 统计文本中的ip地址数量
- 在Bash脚本中 set -e 是什么意思
- 为什么要使用xargs命令
- xargs命令用法实例
- 如何在命令执行超时时自动终止该命令