我有一个这样写的装饰器:
import threading
from time import sleep
from functools import wraps
import sys
import os
def repeat_periodically(f):
""" Repeat wrapped function every second """
@wraps(f)
def wrap(self, *args, **kwargs):
def wrap_helper(*args, **kwargs):
try:
threading.Timer(1.0, wrap_helper).start()
f(self)
except KeyboardInterrupt:
try:
sys.exit(1)
except:
os._exit(1)
wrap_helper()
return wrap
我不确定它是否每次调用自身时都会继续打开一个新线程,但无论如何,当我点击
CTRL + C
时,我无法终止该进程。我还在我装饰的函数中添加了相同的 try-except
块:
@repeat_periodically
def get_stats(self):
try:
# log some state information
except KeyboardInterrupt:
try:
sys.exit(1)
except:
os._exit(1)
我的程序继续运行,我在终端中看到的只是
^C <the stuff that I am logging>
<the stuff that I am logging>
<the stuff that I am logging>
换句话说,它只是不断记录,即使我试图用
CTRL + C
杀死它。
更新:
我应该提到上面的过程是从另一个线程启动的:
tasks = [
{'target': f, 'args': (arg1)},
{'target': g},
]
for task in tasks:
t = threading.Thread(**task)
t.start()
具体来说,是第二个任务启动计时器。但是,如果我设置
t.daemon = True
,该进程只会运行一次并退出。第一个任务使用watchdog。我基本上使用了看门狗文档中的示例代码:
def watch_for_event_file(Event):
path = sys.argv[1] if len(sys.argv) > 1 else '.'
event_handler = LoggingCreateHandler(Event)
observer = Observer()
observer.schedule(event_handler, path)
observer.start()
try:
while True:
time.sleep(1)
except KeyboardInterrupt:
observer.stop()
observer.join()
(抱歉所有更新)
来自线程文档:
The entire Python program exits when no alive non-daemon threads are left.
因此,将
Timer
线程设置为守护线程应该可以解决您的问题。所以替换:
threading.Timer(1.0, wrap_helper).start()
与:
t = threading.Timer(1.0, wrap_helper)
t.daemon = True
t.start()
Ctrl +Shift+ Esc
打开任务管理器
b. 在“进程”或“详细信息”选项卡中查找任何延迟的 Python 进程。
c.如果找到Python进程,右键单击它并选择“结束任务”。完成此操作后,打开 cmd 并再次按 CTRL+C,您将能够终止之前的 python 进程。