https://unix.stackexchange.com/questions/14270/get-exit-status-of-process-thats-piped-to-another
BASH SHELL中,通常使用 $?
来获取上一条命令的返回码。
对于管道中的命令,使用$?
只能获取管道中最后一条命令的返回码,例如
下面的例子中/not/a/valid/filename
是一个不存在的文件
cat /not/a/valid/filename | cat
第一个cat失败,第二个cat成功,所以$?的值为0
这种情况下,可以使用 $PIPESTATUS
来获取管道中每个命令的返回码。
注意:
1、PIPESTATUS 是一个数组,第一条命令的返回码存储在
${PIPESTATUS[0]}
,以此类推,上例中执行完管道中所有的命令后,PIPESTATUS
数组第一个元素值为1,第二个元素值为02、如果前一条命令不是一个管道,而是一个单独的命令,命令的返回码存储为
${PIPESTATUS[0]}
,此时${PIPESTATUS[0]}
同$?值相同(事实上,PIPESTATUS最后一个元素的值总是与$?的值相同)- 3、每执行一条命令,切记PIPESTATUS都会更新其值为上一条命令的返回码,
cat /not/a/valid/filename|cat
if [ ${PIPESTATUS[0]} -ne 0 ]; then echo ${PIPESTATUS[@]}; fi
上例中执行完管道后,`${PIPESTATUS[0]}`值为1,`${PIPESTATUS[1]}`值为0
但是上面的脚本执行完成后,输出为0,这是因为if 分支的测试命令
[root@test output]# cat a.sh
#!/bin/sh
a=3
if [ $a -gt 4 ]
then
echo "1111111"
exit 1
fi
echo "222222"
exit 0
[root@test output]# cat b.sh
#!/bin/sh
sh a.sh | tee >>a.log
aa=${PIPESTATUS[0]}
if [ $aa -ne 0 ]
then
echo "aaaaaaaaaaaaaaaa"
fi