python 发信号杀死主进程,子进程仍然运行的解决方法
在 python 中使用信号来终止进程时,常常会出现这样的问题:发出信号后,主进程终止,但子进程仍在运行。这主要是因为信号的传递范围只限于进程,无法跨进程传递。
解决方案:
- 使用进程组标识(pgid):
进程组是一组由主进程及其所有子进程构成的集合,并且可以与信号一起使用。可以通过以下步骤解决问题:
立即学习“Python免费学习笔记(深入)”;
- 在主进程中,使用 os.getpgid(os.getpid()) 获取进程组 id。
- 将信号处理函数注册到进程组中,而不是主进程。
- 在终止主进程时,使用 os.killpg(pgid, signal.sigterm) 向进程组发送信号。
这样,信号将传递给进程组中的所有进程,包括子进程。
- 将信号处理函数放置在创建子进程之后:
信号处理函数的注册必须放置在创建子进程之后,以确保子进程在注册之前已创建。
修改后的代码示例:
a.py:
import multiprocessing import os import signal import time def process1(): while true: print("子进程运行中") time.sleep(1) def handle_signal(signum, frame): print(signum) if signum == signal.sigterm: print("received sigterm signal. deleting pid file...") try: if os.path.exists("./crawler.pid"): os.remove("./crawler.pid") print("pid file deleted successfully.") except exception as e: print("error deleting pid file:", e) if __name__ == "__main__": a = multiprocessing.process(target=process1) a.daemon = true a.start() child_pid = a.pid parent_pid = os.getpid() parent_pid_group = os.getpgid(parent_pid) signal.signal(signal.sigterm, handle_signal) with open("./crawler.pid", "w") as f: f.write(str(parent_pid_group)) a.join() print(f"父进程pid: {parent_pid}") print(f"子进程pid: {child_pid}")
b.py:
import os import signal with open("./crawler.pid", "r") as f: try: pid = int(f.read()) os.killpg(pid, signal.SIGTERM) print(f"Signal sent successfully to process {pid}") except Exception as e: print(f"Error sending signal: {e}")
以上就是Python子进程不随主进程退出怎么办?的详细内容,更多请关注php中文网其它相关文章!
版权声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系 yyfuon@163.com