Heim > Backend-Entwicklung > Python-Tutorial > Kontextmanager und Daemon-Threads

Kontextmanager und Daemon-Threads

WBOY
Freigeben: 2024-02-06 11:27:08
nach vorne
1030 Leute haben es durchsucht

Kontextmanager und Daemon-Threads

Frageinhalt

Ich starte einen Daemon-Thread von einem Kontextmanager, der jede Sekunde einen Heartbeat senden soll, aber da er in einem Thread läuft, beendet er den Kontextmanager nicht, wenn eine Ausnahme auftritt. Wie löst man im Kontextmanager eine Ausnahme aus, wenn der Heartbeat stoppt?

from contextlib import contextmanager
from threading import Thread, Event
from time import sleep


@contextmanager
def plc():
    stop_event = Event()

    try:
        # Send heartbeat every second
        hb_t = Thread(target=heartbeat_task,
                      args=(stop_event,),
                      daemon=True)
        hb_t.start()

        yield
    except Exception:
        raise
    finally:
        stop_event.set()
        hb_t.join()
        print("Heartbeat stopped")


def heartbeat_task(stop_event):

    value = False
    
    while not stop_event.is_set():

        value = not value

        print("Heartbeat: " + str(value))

        sleep(1)


def main():

    with plc():

        while True:

            print("Program running")

            sleep(5)

if __name__ == '__main__':
    main()
Nach dem Login kopieren

Es fällt mir schwer, Beispiele dafür zu finden.

Danke für deine Hilfe!


Richtige Antwort


aktualisiert

Ich habe den Code geändert, damit er besser mit dem von Ihnen geposteten Code übereinstimmt. Aber:

Der von Ihnen angegebene Code ist inkonsistent: heartbeat_task 传递了一个事件,如果设置该事件将导致函数返回。但只有当使用 with plc(): 创建的函数 main 中的上下文管理器退出时才会设置它,而这是永远不会的。如果您希望 heartbeat_task 抛出的任何异常将强制上下文管理器退出,然后在函数 plc 中捕获,那么调用 stop_event.set() 的意义何在?如果根据定义,我们仅在 heartbeat_task Sind Sie erst hierher gelangt, als er aufgrund einer Ausnahme nicht mehr existiert?

Entweder möchten Sie, dass heartbeat_task 无限期地运行,直到引发异常(在这种情况下,没有“停止”事件的意义),要么您希望能够在存在某些条件时停止 heartbeat_task,但没有这样做的代码。出于演示目的,我假设 main 将有权访问 stop_event 事件,并在某些情况下对其进行设置。否则,它会一直运行,直到检测到 heartbeat_task nie wieder ausgeführt wird, wahrscheinlich weil es eine Ausnahme ausgelöst hat (es führt eine Endlosschleife aus, wie kann es also möglicherweise beendet werden, wenn das Stoppereignis noch nicht festgelegt wurde?). Der Rest ist der Grund, warum Sie einen Kontextmanager verwenden müssen. Ich werde später eine Alternative vorschlagen.

Wenn Sie einen Multithread-Pool verwenden (wir benötigen nur einen Thread im Pool), wird es für den Hauptthread einfacher, die Ausnahme abzufangen, die von der an den Pool übermittelten Aufgabe ausgelöst wird: wenn die multiprocessing.pool.threadpool.apply_async 被调用时返回 multiprocessing.pool.asyncresult 实例,表示未来的完成。当在此实例上调用 get 方法时,您可以从辅助函数 (heartbeat_task) 获取返回值,或者重新引发辅助函数引发的任何异常。但是我们也可以使用 wait 方法来等待提交任务的完成或经过的时间。然后我们可以使用 ready-Methode testet, ob die Aufgabe übermittelt wurde nach 5 Sekunden Wartezeit ist es tatsächlich abgeschlossen (aufgrund einer Ausnahme oder Rückgabe). Wenn die Aufgabe noch läuft, können wir sie anweisen, anzuhalten. In dieser Demo erzwinge ich, dass die Aufgabe nach etwa 7 Sekunden eine Ausnahme auslöst:

from contextlib import contextmanager
from threading import event
from multiprocessing.pool import threadpool
from time import sleep


@contextmanager
def plc():
    stop_event = event()
    pool = threadpool(1)

    # send heartbeat every second
    async_result = pool.apply_async(heartbeat_task, args=(stop_event,))
    yield stop_event, async_result
    # we only return here if the task is no longer running
    try:
        # see if task threw an exception and if so, catch it:
        async_result.get()
    except exception as e:
        print("got exception:", e)
    finally:
        pool.close()
        pool.join()
        print("heartbeat stopped")


def heartbeat_task(stop_event):
    # for demo purposes, we will force an exception to occur
    # after approximately 7 seconds:
    value = false

    n = 0
    while not stop_event.is_set():
        value = not value
        print("heartbeat: " + str(value))
        sleep(1)
        n += 1
        if n == 7:
            raise exception('oops!')


def main():
    with plc() as tpl:
        stop_event, async_result = tpl
        # this function could forcibly cause the heartbeat_task
        # to complete by calling stop_event.set()

        # loop while the task is still running
        while not async_result.ready():
            """
            if some_condition:
                stop_event.set()
                break
            """
            print("program running")
            # sleep for 5 seconds or until heartbeat_task terminates:
            async_result.wait(5)

if __name__ == '__main__':
    main()
Nach dem Login kopieren

Drucken:

program running
heartbeat: true
heartbeat: false
heartbeat: true
heartbeat: false
heartbeat: true
program running
heartbeat: false
heartbeat: true
got exception: oops!
heartbeat stopped
Nach dem Login kopieren

Eine Alternative zur Verwendung von Kontextmanagern

from threading import Event
from multiprocessing.pool import ThreadPool
from time import sleep


def heartbeat_task(stop_event):
    value = False

    n = 0
    while not stop_event.is_set():
        value = not value
        print("Heartbeat: " + str(value))
        sleep(1)
        n += 1
        if n == 7:
            raise Exception('Oops!')

def main():
    stop_event = Event()
    pool = ThreadPool(1)
    async_result = pool.apply_async(heartbeat_task, args=(stop_event,))

    # Run as long as heartbeat_task is running:
    while not async_result.ready():
        """
        if some_condition:
            stop_event.set()
            break
        """
        print("Program running")
        # Sleep for 5 seconds or until heartbeat_task terminates:
        async_result.wait(5)

    # Any exception thrown in heartbeat_task will be rethrown and caught here:
    try:
        async_result.get()
    except Exception as e:
        print("Got exception:", e)
    finally:
        pool.close()
        pool.join()

if __name__ == '__main__':
    main()
Nach dem Login kopieren

Das obige ist der detaillierte Inhalt vonKontextmanager und Daemon-Threads. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:stackoverflow.com
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage