使用python调用外部程序产生进程,优雅关闭的最佳实践是什么?

使用subprocess库:

try:
    p = subprocess.Popen(["ping", "-c", "5", "www.bing.com"], stdout=subprocess.PIPE)
    p.wait()
except KeyboardInterrupt as e:
    p.terminate()

或者使用sh库:

try:
    p = sh.ping("-c", "5", "www.bing.com", _bg=True, _bg_exc=False, 
                _out=sys.stdout, _err=sys.stderr)
    p.wait()
except KeyboardInterrupt as e:
    alive, _ = p.process.is_alive()
    if alive:
        p.process.terminate()

这种结束外部进程的方式有什么缺点?

阅读 3.9k
1 个回答

自问自答下。

我提这个问题的需求是使用python调用外部的第三方程序,类似shell脚本一样。
如果是正常的多进程编程协作完成一个任务,那么当然最好的方式是两个进程协作,即使用IPC的方式使父进程传递给子进程结束的信息,然后子进程自己优雅的退出。

而如果使用python调用外部的第三方命令,将其作为类似shell脚本的方式使用,由于无法修改外部程序的代码,因此IPC的手段非常有限,仅仅限于向子进程发送信号。

那么该问题的思路就很清晰了,我之前有一次是使用terminate()函数结束子进程导致了子进程无法正常释放资源(linux系统)。通过查询文档可知terminate()函数在linux系统下的行为是向进程发送SIGTERM。那么,问题的原因则是该第三方程序接收到SIGTERM信号后的处理逻辑有问题。

因此,结束该进程最好的方式是使用该第三方程序能够接受的方式,这通过查阅该第三方程序的文档可以得到。

比如说,命令行程序对SIGINT信号的考虑更完善,因为在终端使用时按Ctrl-C发送的就是SIGINT信号。因此,对于大部分的第三方程序,发送SIGINT信号更好:

p.process.signal(signal.SIGINT)    
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题