Linux系统下我们调用脚本一般有两个命令可以使用:system和popen。system命令无法获取脚本执行的输出结果,但是popen命令通过读的方式执行shell脚本则可以获得脚本执行的输出结果。
一般来说,如上图所示,popen的使用步骤为popen执行命令,然后通过返回的句柄循环获取输出结果,最后关闭句柄。这个没有任何问题。但是如果我们没有循环通过fgets获取输出就直接调用pclose关闭句柄会发生什么事情呢?
pclose的核心工作是关闭文件句柄,如果popen以读的方式执行,则关闭的是标准输出的管道,如果以写的方式执行,则关闭的是标准输入的管道。关闭完成之后,它会等待脚本执行完毕,看起来仍然非常完美。但是脚本中的很多命令比如echo的执行需要依赖这些标准输入或者标准输出管道,当这些管道被关闭时会触发broken pipe,最终导致脚本异常结束。
综上所述,在脚本没有完全执行完毕的时候调用pclose可能会导致意想不到的问题,简单的解决办法是通过循环调用fgets这样的函数获取输出结果来等待脚本完全执行完毕,或者不需要返回结果的时候直接使用system来替代popen。