shell 管道和tee使用时获取前面命令返回值

时间:2022-03-07 15:09:42

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,第二个元素值为0

  • 2、如果前一条命令不是一个管道,而是一个单独的命令,命令的返回码存储为${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