我怎样才能实现进度条 in a 弹出窗口它通过所谓的 Worker 类(即消耗时间/CPU 的任务)监视正在运行的函数的进度QThread?
我已经检查了无数的示例和教程,但进度条显示在弹出窗口中的事实似乎让一切变得更加困难。我相信我想要的是一件相当简单的事情,但我一直失败,而且我没有想法。
我有一个我想要实现的目标的例子,它基于这个答案:
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 中),程序崩溃,并且我没有收到任何明确的错误消息。
When I debug it though, it looks like it works, as I am able to see what I intended to achieve:
![App working during debugging](https://i.stack.imgur.com/g25Ag.png)
我有一系列的问题:
- 为什么会崩溃?
- 为什么在调试过程中会起作用?
- 我是否应该放弃并实施进度条(锚定)
应用程序的主窗口?
- 我过去已经实现了类似的事情,但没有线程:在工作函数(即 CPU 消耗函数)的循环中,我必须添加
QApplication.processEvents()
这样在每次迭代时进度条都会有效更新。以这种方式做事显然不是最理想的。与我现在想要实现的目标相比,它仍然是更好的选择吗?
如果我遗漏了一些明显的东西,或者如果这个问题已经在某个地方得到了回答(重复),请原谅:我无法找到这个问题的答案。预先非常感谢您。