3

我认为 Python 进程在终止时会调用它们的 atexit 函数。请注意,我使用的是 Python 2.7。这是一个简单的例子:

from __future__ import print_function
import atexit
from multiprocessing import Process


def test():
    atexit.register(lambda: print("atexit function ran"))

process = Process(target=test)
process.start()
process.join()

我希望这会打印“atexit function run”,但事实并非如此。

注意这个问题: Python process won't call atexit 是类似的,但它涉及以信号终止的进程,答案涉及拦截该信号。这个问题中的进程正在优雅地退出,因此(据我所知)该问题和答案不适用(除非这些进程由于某种信号而退出?)。

4

1 回答 1

3

我通过研究 CPython 中的实现方式做了一些研究。这是假设您在 Unix 上运行。如果您在 Windows 上运行,则以下可能无效,因为进程的实现multiprocessing不同。

事实证明,os._exit()总是在过程结束时调用。这与atexit文档中的以下注释一起,应该可以解释为什么您的 lambda 没有运行。

注意:当程序被 Python 未处理的信号杀死,检测到 Python 致命内部错误或调用 os._exit() 时,不会调用通过此模块注册的函数。


这是 CPython 2.7 类的摘录Popen,用于分叉进程。请注意,分叉进程的最后一条语句是调用os._exit().

# Lib/multiprocessing/forking.py

class Popen(object):

    def __init__(self, process_obj):
        sys.stdout.flush()
        sys.stderr.flush()
        self.returncode = None

        self.pid = os.fork()
        if self.pid == 0:
            if 'random' in sys.modules:
                import random
                random.seed()
            code = process_obj._bootstrap()
            sys.stdout.flush()
            sys.stderr.flush()
            os._exit(code)

在 Python 3.4 中,os._exit()如果您正在启动分叉进程,它仍然存在,这是默认设置。但似乎您可以更改它,请参阅上下文和启动方法以获取更多信息。我还没有尝试过,但也许使用spawn的 start 方法会起作用?但不适用于 Python 2.7。

于 2014-10-20T23:28:00.857 回答