- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
为什么线程应该持续存在并阻止其进程退出,即使在其目标完成之后也是如此?
While this question uses an additional child-process, the underlaying issue is entirely rooted in multithreading. Therefore this basic issue can be reproduced with the
MainProcess
alone. (Edited by @Darkonaut)
我创建了一个继承multiprocessing.Process
的类:
class Task(Process):
def run(self) :
print("RUN")
t = threading.Thread(target=do_some_work)
t.start()
# ...
t.join()
print("CLOSED")
我是这样开始的:
proc = Task()
proc.start()
proc.join()
print("JOINED")
但是它不会加入,输出将是这样的:
>> RUN
>> CLOSED
我没有使用任何类型的队列
和管道
。
当我在 Ubuntu 上运行这个程序时,我用它的 pid 跟踪了该进程。即使在 print("CLOSED")
行完成后,该进程仍然存在,没有任何异常。我还在 Windows 上运行了这个程序并在任务管理器中跟踪了该过程。进程在 print("CLOSED")
后退出,但仍未加入。
另一点是,在 Ubuntu 上,当 print("CLOSED")
之后一切都卡住并且我按 Ctrl + C
时,我得到:
Traceback (most recent call last):
File "Scheduler.py", line 164, in <module>
scheduler.start()
File "Scheduler.py", line 152, in start
self.enqueueTask(plan)
File "Scheduler.py", line 134, in enqueueTask
proc.join()
File "/usr/local/lib/python3.8/multiprocessing/process.py", line 149, in join
res = self._popen.wait(timeout)
File "/usr/local/lib/python3.8/multiprocessing/popen_fork.py", line 47, in wait
return self.poll(os.WNOHANG if timeout == 0.0 else 0)
File "/usr/local/lib/python3.8/multiprocessing/popen_fork.py", line 27, in poll
pid, sts = os.waitpid(self.pid, flag)
根据最后一行,我猜主进程正在等待某些东西,但是什么以及为什么?
问题似乎出在我在 Task
的 run()
方法中启动的非守护线程。 使该线程成为守护线程可以解决问题,因此我可以肯定地说,即使在其 MainThread
完成后,该线程也会阻止我的进程关闭。我仍然很困惑,因为该非守护线程的目标函数已成功完成。
最佳答案
Why should a thread persist and prevent its process to exit, even after its target is done?
虽然这个问题使用了额外的子进程,但底层问题完全源于多线程。因此,这个基本问题可以通过 MainProcess
重现。独自的。涉及额外子进程的答案可以在 edit 2 中找到。 .
在没有看到子进程中的新线程真正在做什么的情况下,您观察到的行为的可能情况是您的 thread-1
正在开始又一个 thread-2
,你甚至可能不知道。可能它是从您正在调用的第三方库启动的,或者保留在 stdlib 中,multiprocessing.Queue.put()
还在后台启动一个供给线程。
这个一般场景不是 Process
-子类化问题与调用Process.close()
无关来自子进程本身(不正确的使用,但没有后果)。
MainThread
进程中始终是退出进程中的最后一个线程,并且它正在加入非守护线程作为其 _shutdown()
的一部分-常规。这就是保持 MainThread
的原因。处于不确定状态,而其“表面”工作已经完成。
The problem is with a non-daemon thread that I'm starting in run() method of Task. so I can surely say that thread is preventing my process to be closed even after its
MainThread
is done. but I'm still confused because target function of that non-daemon thread is done successfully.
现在,在如图所示的场景中,您的目标函数为 thread-1
实际上可以成功完成。然而这个thread-1
已开始另一个thread-2
,然后它会执行一些持续很长时间的操作,例如在最坏的情况下永远阻塞。
Q: If
thread-1
itself is not the problem, why there is no hanging when you makethread-1
adaemon
?
这是因为守护进程标志的 "initial value is inherited from the creating thread" 。所以制作thread-1
一个daemon
,使其后代 thread-2
一个daemon
也是如此,除非daemon
-标志 thread-2
被明确设置。守护进程在关闭时不会加入,整个过程“当没有剩余的事件非守护线程时退出”。
请注意,在 Python 3.7 之前,由 Process
创建的非守护线程尚未加入。MainProcess
之外的线程的这种不同行为已修复于 bpo-18966 .
为了表明这种情况已经可以通过更简单的设置重现,下面的示例使用 MainProcess
作为不会退出的进程。 thread-2
这是 Timer
-thread,它将启动并调用 threading.Barrier(parties=1).wait()
10秒后。这个.wait()
然后调用将立即结束 parties=1
,或使用 parties=2
永远阻止因为没有其他人调用.wait()
关于这个 Barrier
存在于我们的设置中。这使得我们能够轻松切换我们想要重现的行为。
import threading
def blackbox(parties):
"""Dummy for starting thread we might not know about."""
timer = threading.Timer(10, threading.Barrier(parties=parties).wait) # Thread-2
timer.name = "TimerThread"
timer.start()
def t1_target(parties): # Thread-1
"""Start another thread and exit without joining."""
logger = get_mp_logger()
logger.info(f"ALIVE: {[t.name for t in threading.enumerate()]}")
blackbox(parties)
logger.info(f"ALIVE: {[t.name for t in threading.enumerate()]}")
logger.info("DONE")
if __name__ == '__main__':
import logging
parties = 1
daemon = False
print(f"parties={parties}, daemon={daemon}")
logger = get_mp_logger(logging.INFO)
logger.info(f"ALIVE: {[t.name for t in threading.enumerate()]}")
t = threading.Thread(target=t1_target, args=(parties,), daemon=daemon)
t.start()
t.join()
logger.info(f"ALIVE: {[t.name for t in threading.enumerate()]}")
logger.info("DONE")
下面的日志适用于parties=1
,所以不存在无限阻塞,但是因为 thread-2
不是守护线程,MainThread
将在关机时加入它。请注意TimerThread
t1_target
之后仍然活着已经完成了。这里主要感兴趣的是 MainThread
如何从 "DONE"
需要大约 10 秒至"process shutting down"
。这是 10 秒 TimerThread
还活着。
parties=1, daemon=False
[18:04:31,977 MainThread <module>] ALIVE: ['MainThread']
[18:04:31,977 Thread-1 t1_target] ALIVE: ['MainThread', 'Thread-1']
[18:04:31,978 Thread-1 t1_target] ALIVE: ['MainThread', 'Thread-1', 'TimerThread']
[18:04:31,978 Thread-1 t1_target] DONE
[18:04:31,978 MainThread <module>] ALIVE: ['MainThread', 'TimerThread']
[18:04:31,978 MainThread <module>] DONE
[18:04:41,978 MainThread info] process shutting down
Process finished with exit code 0
与 parties=2
它在这个阶段永远挂起,...
parties=2, daemon=False
[18:05:06,010 MainThread <module>] ALIVE: ['MainThread']
[18:05:06,010 Thread-1 t1_target] ALIVE: ['MainThread', 'Thread-1']
[18:05:06,011 Thread-1 t1_target] ALIVE: ['MainThread', 'Thread-1', 'TimerThread']
[18:05:06,011 Thread-1 t1_target] DONE
[18:05:06,011 MainThread <module>] ALIVE: ['MainThread', 'TimerThread']
[18:05:06,011 MainThread <module>] DONE
...除非您还设置了 daemon=True
,或者 thread-1
( thread-2
继承)或仅用于 thread-2
直接。
parties=2, daemon=True
[18:05:35,539 MainThread <module>] ALIVE: ['MainThread']
[18:05:35,539 Thread-1 t1_target] ALIVE: ['MainThread', 'Thread-1']
[18:05:35,539 Thread-1 t1_target] ALIVE: ['MainThread', 'Thread-1', 'TimerThread']
[18:05:35,539 Thread-1 t1_target] DONE
[18:05:35,539 MainThread <module>] ALIVE: ['MainThread', 'TimerThread']
[18:05:35,539 MainThread <module>] DONE
[18:05:35,539 MainThread info] process shutting down
Process finished with exit code 0
<小时/>
helper
DEFAULT_MP_FORMAT = \
'[%(asctime)s,%(msecs)03d %(threadName)s %(funcName)s]' \
' %(message)s'
DEFAULT_DATEFORMAT = "%H:%M:%S" # "%Y-%m-%d %H:%M:%S"
def get_mp_logger(level=None, fmt=DEFAULT_MP_FORMAT, datefmt=DEFAULT_DATEFORMAT):
"""
Initialize multiprocessing-logger if needed and return reference.
"""
import multiprocessing.util as util
import logging
logger = util.get_logger()
if not logger.handlers:
logger = util.log_to_stderr(level)
logger.handlers[0].setFormatter(logging.Formatter(fmt, datefmt))
return logger
关于python - 线程已完成的进程永远不会退出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59796473/
我目前正在为 Cocoa/Objective-C 项目编写一个脚本来完全自动化我的编译-运行-调试过程。 我的最后一行代码是: lldb -f Build/MyApp.app -o "run" 这实际
我有一个带有登录屏幕的脚本,如果按下取消按钮,我想完全退出该应用程序。我尝试了 3 种方法: 系统退出() QApplication.quit() QCoreApplication.instance(
我有一个 Flash 应用程序,可以重定向到另一个页面。我很乐意捕获任何其他窗口卸载事件(单击链接/提交表单)并警告用户他们将丢失 Flash 应用程序中的进度。 但是,我找不到任何方法来判断 URL
我正在尝试在 Ubuntu 上用 Python 编写一个简单的程序,它将在播放视频完成后关闭/退出/退出 VLC Player。 能否请您指导我应该在我的程序中添加什么以获得我需要的结果。 impor
我在 Lynda.com 上学习 PHP 2 视频时遇到了一个问题,因为讲师似乎忽略了告诉我们他在视频中执行的步骤之一。我在这里上传了相关视频http://www.youtube.com/watch?
某天在群内有同学问到,在python下我用input或者raw_input都得输入完后回车才能获取到输入的值,那如何实现任意键退出暂停等功能呢,我当时也没有多想,因为接触python时间也不算长,主
我按顺序调用了几个函数(我无法编辑),但有些函数会重定向用户,所以我永远不会进入下一个函数。 我正在调用一个第三方函数,它调用了我能够阻止的 wp_redirect(),但是下一行是 exit;我不知
终止/退出主函数的 D 方式是什么? import std.stdio; import core.thread; void main() { int i; while (i <= 5)
我正在申请写作。用户可以打开应用程序、写一些文本、保存他们的工作等。 我试图做到这一点,以便点击 window close按钮将提示用户 (a) 保存他们的工作(如有必要)或 (b) 退出。 我正在尝
我正在通过在 repl 中检查别人的代码来玩弄它。 它不断调用 System/exit,这导致我的 repl 崩溃。这真是令人气愤。 在我有权访问的所有代码中,我都模拟了调用。 但它也会调用一些我没有
我正在使用 subprocess执行mimic的模块程序(指定 here )。下面的代码成功地读取了一些文本并写入了一个 mp3 文件。 import subprocess proc = subpro
退出 .then 范围后数组上的值被清除 在下面的代码中tableValues1.length 给我正确的长度,直到它位于每个循环内当它退出时,作用域数组长度为零。 请谁能帮我解决这个问题 - 谢谢
我正在尝试为 s3cmd 编写一个 docker 镜像。当我通过 docker-compose 运行从 Dockerfile 构建的图像时,容器在 docker compose run 命令之前退出。
这个问题已经有答案了: 已关闭12 年前。 Possible Duplicate: Quitting an application - is that frowned upon? 我编写了一个 And
我遇到 Selenium WebDriver 仅执行一次后退出 for 循环的问题。据推测,这是获取内容和在加载页面之前启动循环的问题。是否有可能让 webdriver 等待页面加载? List al
#include #include #include #include "Player.h" using namespace std; void PlayerMenu(); int main()
class Test{ public static void main(String args[]) { Patron list[] = new PatronData(
我正在做一些作业,遇到了这个问题。 Write a program that reads several lines of text and prints a table indicating the
我正在用 C 创建一个简单的 Linux 命令 shell。我无法理解我的代码在哪里出现问题。 “commands”是我希望作为一个父进程的子进程同时执行的 Linux 命令字符串列表。当所有执行完成
我的控制台应用程序有点问题。应用程序应该从用户那里获取数字并将它们添加到列表中,但是如果输入是“c”,它应该关闭。我不知道如何在不使用 Scanner.nextLine() 挂起应用程序并退出循环的情
我是一名优秀的程序员,十分优秀!