乐闻世界logo
搜索文章和话题

How to terminate a python subprocess launched with shell= True

5 个月前提问
4 个月前修改
浏览次数22

1个答案

1

确保子进程能够安全、有效地终止,通常有几种方法可以实现。当使用shell=True参数通过Python的subprocess模块启动子进程时,情况会稍微复杂一些,因为此时实际上是启动了一个shell进程,这个shell又去启动了指定的命令。以下是一些终止这类子进程的常用方法:

方法1:使用Popen对象的terminate()方法

这是最直接的方法。terminate()将向进程发送SIGTERM信号。这种方法适用于大多数UNIX系统。在Windows上,它将调用TerminateProcess()来结束进程。

示例代码:

python
import subprocess import time # 启动一个子进程,使用shell=True p = subprocess.Popen("some_long_running_command", shell=True) time.sleep(5) # 假设我们5秒后想要终止它 # 终止进程 p.terminate() # 等待进程终止 p.wait() print("进程已终止")

方法2:发送特定的信号

如果terminate()方法不能满足需求,或者需要更精细的控制,可以考虑使用send_signal()方法发送特定的信号。例如,在UNIX系统上,可以发送SIGKILL信号来强制终止进程。

示例代码:

python
import subprocess import signal import time p = subprocess.Popen("some_long_running_command", shell=True) time.sleep(5) # 发送SIGKILL信号,UNIX系统有效 p.send_signal(signal.SIGKILL) # 等待进程终止 p.wait() print("进程已终止")

方法3:杀死整个进程组

使用shell=True时,subprocess.Popen会创建一个新的shell作为子进程,而该命令可能会启动更多的进程。在这种情况下,单独终止shell可能不足以停止所有子进程。此时可以考虑终止整个进程组。

示例代码:

python
import subprocess import os import signal import time # 使用os.setsid启动新的会话 p = subprocess.Popen("some_long_running_command", shell=True, preexec_fn=os.setsid) time.sleep(5) # 发送SIGTERM信号到整个进程组 os.killpg(os.getpgid(p.pid), signal.SIGTERM) # 等待进程终止 p.wait() print("进程组已终止")

注意事项

使用shell=True时应格外小心,因为它可能会增加安全风险,尤其是当传入的命令包括来自不可靠来源的输入时。始终尽可能避免使用shell=True,或者确保传入的命令是经过严格验证的。

在实际应用中,需要根据具体的操作系统和需求选择最合适的方法。在开发跨平台的应用时,还需要考虑到不同操作系统间的差异。

2024年8月16日 23:23 回复

你的答案