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()
Es fällt mir schwer, Beispiele dafür zu finden.
Danke für deine Hilfe!
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()
Drucken:
program running heartbeat: true heartbeat: false heartbeat: true heartbeat: false heartbeat: true program running heartbeat: false heartbeat: true got exception: oops! heartbeat stopped
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()
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!