PyQt5:使用QThread弹出进度条
PyQt5: pop-up progressbar using QThread
如何在 弹出窗口 window 中实现 进度条 以监控 运行ning 函数来自所谓的 Worker class(即 time/CPU-consuming 任务)通过 QThread?
我检查了无数示例和教程,但进度条显示在弹出窗口中的事实 window 似乎让一切变得更难了。我相信我想要的是一件相当简单的事情,但我总是失败,我 运行 没有想法。
我有一个我想要实现的例子,它基于 this answer:
import sys
import time
from PyQt5.QtCore import QThread, pyqtSignal, QObject, pyqtSlot
from PyQt5.QtWidgets import QApplication, QPushButton, QWidget, QHBoxLayout, QProgressBar, QVBoxLayout
class MainWindow(QWidget):
def __init__(self):
super().__init__()
self.setWindowTitle("Widget")
self.h_box = QHBoxLayout(self)
self.main_window_button = QPushButton("Start")
self.main_window_button.clicked.connect(PopUpProgressB)
self.h_box.addWidget(self.main_window_button)
self.setLayout(self.h_box)
self.show()
class Worker(QObject):
finished = pyqtSignal()
intReady = pyqtSignal(int)
@pyqtSlot()
def proc_counter(self): # A slot takes no params
for i in range(1, 100):
time.sleep(1)
self.intReady.emit(i)
self.finished.emit()
class PopUpProgressB(QWidget):
def __init__(self):
super().__init__()
self.pbar = QProgressBar(self)
self.pbar.setGeometry(30, 40, 500, 75)
self.layout = QVBoxLayout()
self.layout.addWidget(self.pbar)
self.setLayout(self.layout)
self.setGeometry(300, 300, 550, 100)
self.setWindowTitle('Progress Bar')
self.show()
self.obj = Worker()
self.thread = QThread()
self.obj.intReady.connect(self.on_count_changed)
self.obj.moveToThread(self.thread)
self.obj.finished.connect(self.thread.quit)
self.thread.started.connect(self.obj.proc_counter)
self.thread.start()
def on_count_changed(self, value):
self.pbar.setValue(value)
if __name__ == '__main__':
app = QApplication(sys.argv)
main_window = MainWindow()
sys.exit(app.exec_())
当我 运行 后者时(例如在 PyCharm Community 2019.3 中),程序崩溃并且我没有收到任何明确的错误消息。
不过,当我调试它时,它看起来好像可以工作,因为我能够看到我打算实现的目标:
我有一系列问题:
- 为什么会崩溃?
- 为什么它在调试时起作用?
- 我是否可以放弃并在中实现进度条(锚定)
应用程序的主要 window?
- 我过去已经实现了类似的东西但没有线程:在工作函数的循环中(即 CPU 消耗函数)我必须添加
QApplication.processEvents()
以便在每次迭代时进度条已有效更新。以这种方式做事显然不是最优的。它仍然是我现在想要实现的更好的替代方案吗?
如果我遗漏了一些明显的东西,或者如果已经在某个地方回答了这个问题(重复),请原谅:我无法找到这个问题的答案。非常感谢您。
解释:
要理解问题,您必须了解以下内容:
self.main_window_button.clicked.connect(PopUpProgressB)
相当于:
self.main_window_button.clicked.connect(foo)
# ...
def foo():
PopUpProgressB()
观察到按下按钮时创建的 PopUpProgressB 对象没有生命周期,就像 "foo" 函数的执行几乎是瞬时的,因此弹出窗口将显示和隐藏在很短的时间内。
解决方案:
这个想法是弹出窗口有一个范围,允许它有一个足够大的生命周期来显示它应该在 class 属性弹出对象上取得的进展。
# ...
self.main_window_button = QPushButton("Start")
<b>self.popup = PopUpProgressB()
self.main_window_button.clicked.connect(self.popup.show)</b>
self.h_box.addWidget(self.main_window_button)
# ...
为了不显示,您必须删除对 PopUpProgressB 的 show() 方法的调用:
class PopUpProgressB(QWidget):
def __init__(self):
super().__init__()
# ...
self.setWindowTitle('Progress Bar')
<b># self.show() # <--- remove this line</b>
self.obj = Worker()
# ...
既然我已经解释了你问题的失败,我会回答你的问题:
为什么会崩溃?删除popup对象时,创建的QThread也被删除但是Qt访问不再分配内存(core dumped)导致应用程序在不抛出任何异常的情况下关闭。
为什么它在调试时起作用?很多像PyCharm这样的IDE不处理Qt错误,所以恕我直言,当他们有这样的错误时建议他们在terminal/CMD中执行他们的代码,例如当我执行你的代码时我得到:
QThread: Destroyed while thread is still running
Aborted (core dumped)
是否应该放弃并在应用程序的主要 window 中实现进度条(锚定)? 否。
我过去已经实现了类似的东西但没有线程:在工作函数的循环中(即 CPU-消费函数)我不得不添加QApplication.processEvents() 以便在每次迭代中有效更新进度条。以这种方式做事显然不是最优的。它仍然是我现在想要实现的更好的替代方案吗? 如果有更好的替代方案,请不要使用 QApplication::processEvents(),在这种情况下,线程是最好的,因为它成为主线程不太忙。
最后,初学者在Qt中报的很多错误都与变量的范围有关,所以我建议大家分析一下每个变量应该有多大,比如你想让一个对象与 class 一样生活,然后使该变量成为 class 的属性,如果您只在方法中使用它,那么它只是一个局部变量,等等
基于 ,这是一个工作代码的样子:
import sys
import time
from PyQt5.QtCore import QThread, pyqtSignal, QObject, pyqtSlot
from PyQt5.QtWidgets import QApplication, QPushButton, QWidget, QHBoxLayout, QProgressBar, QVBoxLayout
class MainWindow(QWidget):
def __init__(self):
super().__init__()
self.setWindowTitle("Widget")
self.h_box = QHBoxLayout(self)
self.main_window_button = QPushButton("Start")
self.popup = PopUpProgressB() # Creating an instance instead as an attribute instead of creating one
# everytime the button is pressed
self.main_window_button.clicked.connect(self.popup.start_progress) # To (re)start the progress
self.h_box.addWidget(self.main_window_button)
self.setLayout(self.h_box)
self.show()
class Worker(QObject):
finished = pyqtSignal()
intReady = pyqtSignal(int)
@pyqtSlot()
def proc_counter(self): # A slot takes no params
for i in range(1, 100):
time.sleep(0.1)
self.intReady.emit(i)
self.finished.emit()
class PopUpProgressB(QWidget):
def __init__(self):
super().__init__()
self.pbar = QProgressBar(self)
self.pbar.setGeometry(30, 40, 500, 75)
self.layout = QVBoxLayout()
self.layout.addWidget(self.pbar)
self.setLayout(self.layout)
self.setGeometry(300, 300, 550, 100)
self.setWindowTitle('Progress Bar')
# self.show()
self.obj = Worker()
self.thread = QThread()
self.obj.intReady.connect(self.on_count_changed)
self.obj.moveToThread(self.thread)
self.obj.finished.connect(self.thread.quit)
self.obj.finished.connect(self.hide) # To hide the progress bar after the progress is completed
self.thread.started.connect(self.obj.proc_counter)
# self.thread.start() # This was moved to start_progress
def start_progress(self): # To restart the progress every time
self.show()
self.thread.start()
def on_count_changed(self, value):
self.pbar.setValue(value)
if __name__ == '__main__':
app = QApplication(sys.argv)
main_window = MainWindow()
sys.exit(app.exec_())
如何在 弹出窗口 window 中实现 进度条 以监控 运行ning 函数来自所谓的 Worker class(即 time/CPU-consuming 任务)通过 QThread?
我检查了无数示例和教程,但进度条显示在弹出窗口中的事实 window 似乎让一切变得更难了。我相信我想要的是一件相当简单的事情,但我总是失败,我 运行 没有想法。
我有一个我想要实现的例子,它基于 this answer:
import sys
import time
from PyQt5.QtCore import QThread, pyqtSignal, QObject, pyqtSlot
from PyQt5.QtWidgets import QApplication, QPushButton, QWidget, QHBoxLayout, QProgressBar, QVBoxLayout
class MainWindow(QWidget):
def __init__(self):
super().__init__()
self.setWindowTitle("Widget")
self.h_box = QHBoxLayout(self)
self.main_window_button = QPushButton("Start")
self.main_window_button.clicked.connect(PopUpProgressB)
self.h_box.addWidget(self.main_window_button)
self.setLayout(self.h_box)
self.show()
class Worker(QObject):
finished = pyqtSignal()
intReady = pyqtSignal(int)
@pyqtSlot()
def proc_counter(self): # A slot takes no params
for i in range(1, 100):
time.sleep(1)
self.intReady.emit(i)
self.finished.emit()
class PopUpProgressB(QWidget):
def __init__(self):
super().__init__()
self.pbar = QProgressBar(self)
self.pbar.setGeometry(30, 40, 500, 75)
self.layout = QVBoxLayout()
self.layout.addWidget(self.pbar)
self.setLayout(self.layout)
self.setGeometry(300, 300, 550, 100)
self.setWindowTitle('Progress Bar')
self.show()
self.obj = Worker()
self.thread = QThread()
self.obj.intReady.connect(self.on_count_changed)
self.obj.moveToThread(self.thread)
self.obj.finished.connect(self.thread.quit)
self.thread.started.connect(self.obj.proc_counter)
self.thread.start()
def on_count_changed(self, value):
self.pbar.setValue(value)
if __name__ == '__main__':
app = QApplication(sys.argv)
main_window = MainWindow()
sys.exit(app.exec_())
当我 运行 后者时(例如在 PyCharm Community 2019.3 中),程序崩溃并且我没有收到任何明确的错误消息。
不过,当我调试它时,它看起来好像可以工作,因为我能够看到我打算实现的目标:
我有一系列问题:
- 为什么会崩溃?
- 为什么它在调试时起作用?
- 我是否可以放弃并在中实现进度条(锚定) 应用程序的主要 window?
- 我过去已经实现了类似的东西但没有线程:在工作函数的循环中(即 CPU 消耗函数)我必须添加
QApplication.processEvents()
以便在每次迭代时进度条已有效更新。以这种方式做事显然不是最优的。它仍然是我现在想要实现的更好的替代方案吗?
如果我遗漏了一些明显的东西,或者如果已经在某个地方回答了这个问题(重复),请原谅:我无法找到这个问题的答案。非常感谢您。
解释:
要理解问题,您必须了解以下内容:
self.main_window_button.clicked.connect(PopUpProgressB)
相当于:
self.main_window_button.clicked.connect(foo)
# ...
def foo():
PopUpProgressB()
观察到按下按钮时创建的 PopUpProgressB 对象没有生命周期,就像 "foo" 函数的执行几乎是瞬时的,因此弹出窗口将显示和隐藏在很短的时间内。
解决方案:
这个想法是弹出窗口有一个范围,允许它有一个足够大的生命周期来显示它应该在 class 属性弹出对象上取得的进展。
# ...
self.main_window_button = QPushButton("Start")
<b>self.popup = PopUpProgressB()
self.main_window_button.clicked.connect(self.popup.show)</b>
self.h_box.addWidget(self.main_window_button)
# ...
为了不显示,您必须删除对 PopUpProgressB 的 show() 方法的调用:
class PopUpProgressB(QWidget):
def __init__(self):
super().__init__()
# ...
self.setWindowTitle('Progress Bar')
<b># self.show() # <--- remove this line</b>
self.obj = Worker()
# ...
既然我已经解释了你问题的失败,我会回答你的问题:
为什么会崩溃?删除popup对象时,创建的QThread也被删除但是Qt访问不再分配内存(core dumped)导致应用程序在不抛出任何异常的情况下关闭。
为什么它在调试时起作用?很多像PyCharm这样的IDE不处理Qt错误,所以恕我直言,当他们有这样的错误时建议他们在terminal/CMD中执行他们的代码,例如当我执行你的代码时我得到:
QThread: Destroyed while thread is still running Aborted (core dumped)
是否应该放弃并在应用程序的主要 window 中实现进度条(锚定)? 否。
我过去已经实现了类似的东西但没有线程:在工作函数的循环中(即 CPU-消费函数)我不得不添加QApplication.processEvents() 以便在每次迭代中有效更新进度条。以这种方式做事显然不是最优的。它仍然是我现在想要实现的更好的替代方案吗? 如果有更好的替代方案,请不要使用 QApplication::processEvents(),在这种情况下,线程是最好的,因为它成为主线程不太忙。
最后,初学者在Qt中报的很多错误都与变量的范围有关,所以我建议大家分析一下每个变量应该有多大,比如你想让一个对象与 class 一样生活,然后使该变量成为 class 的属性,如果您只在方法中使用它,那么它只是一个局部变量,等等
基于
import sys
import time
from PyQt5.QtCore import QThread, pyqtSignal, QObject, pyqtSlot
from PyQt5.QtWidgets import QApplication, QPushButton, QWidget, QHBoxLayout, QProgressBar, QVBoxLayout
class MainWindow(QWidget):
def __init__(self):
super().__init__()
self.setWindowTitle("Widget")
self.h_box = QHBoxLayout(self)
self.main_window_button = QPushButton("Start")
self.popup = PopUpProgressB() # Creating an instance instead as an attribute instead of creating one
# everytime the button is pressed
self.main_window_button.clicked.connect(self.popup.start_progress) # To (re)start the progress
self.h_box.addWidget(self.main_window_button)
self.setLayout(self.h_box)
self.show()
class Worker(QObject):
finished = pyqtSignal()
intReady = pyqtSignal(int)
@pyqtSlot()
def proc_counter(self): # A slot takes no params
for i in range(1, 100):
time.sleep(0.1)
self.intReady.emit(i)
self.finished.emit()
class PopUpProgressB(QWidget):
def __init__(self):
super().__init__()
self.pbar = QProgressBar(self)
self.pbar.setGeometry(30, 40, 500, 75)
self.layout = QVBoxLayout()
self.layout.addWidget(self.pbar)
self.setLayout(self.layout)
self.setGeometry(300, 300, 550, 100)
self.setWindowTitle('Progress Bar')
# self.show()
self.obj = Worker()
self.thread = QThread()
self.obj.intReady.connect(self.on_count_changed)
self.obj.moveToThread(self.thread)
self.obj.finished.connect(self.thread.quit)
self.obj.finished.connect(self.hide) # To hide the progress bar after the progress is completed
self.thread.started.connect(self.obj.proc_counter)
# self.thread.start() # This was moved to start_progress
def start_progress(self): # To restart the progress every time
self.show()
self.thread.start()
def on_count_changed(self, value):
self.pbar.setValue(value)
if __name__ == '__main__':
app = QApplication(sys.argv)
main_window = MainWindow()
sys.exit(app.exec_())