python - PyQt5:使用QThread弹出进度条

标签 python pyqt pyqt5 qthread qprogressbar

如何在弹出窗口中实现一个进度条,用于监视所谓的 Worker 类中正在运行的函数的进度(即时间/CPU -消耗任务)通过QThread

我已经检查了无数的示例和教程,但进度条显示在弹出窗口中的事实似乎让一切变得更加困难。我相信我想要的是一件相当简单的事情,但我一直失败,而且我没有想法。

我有一个我想要实现的目标的示例,它基于 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 中),程序崩溃,并且我没有收到任何明确的错误消息。

当我调试它时,它看起来像是有效的,因为我能够看到我想要实现的目标: App working during debugging

我有一系列问题:

  1. 为什么会崩溃?
  2. 为什么它在调试时有效?
  3. 我是否可以放弃并实现进度条(锚定) 应用程序的主窗口?
  4. 我过去已经实现了类似的事情,但没有线程:在工作函数(即 CPU 消耗函数)的循环中,我必须添加 QApplication.processEvents() 以便在每个迭代进度条被有效更新。以这种方式做事显然不是最理想的。与我现在想要实现的目标相比,它仍然是更好的选择吗?

如果我明显遗漏了一些东西,或者如果这个问题已经在某个地方得到了回答(重复),请原谅:我无法找到这个问题的答案。预先非常感谢您。

最佳答案

说明:

要理解这个问题,您必须了解以下内容:

self.main_window_button.clicked.connect(PopUpProgressB)

相当于:

self.main_window_button.clicked.connect(foo)
# ...
def foo():
    PopUpProgressB()

观察到,当按下按钮时,会创建一个 PopUpProgressB 对象,该对象没有生命周期,就像“foo”函数的执行一样,几乎是瞬时的,因此弹出窗口将在很短的时间内显示和隐藏时间。

解决方案:

这个想法是,弹出窗口有一个范围,允许它有一个足够大的生命周期,以显示它应该对类属性弹出对象进行的进度。

# ...
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()
        # ...
<小时/>

既然我已经解释了你的问题的失败,我将回答你的问题:

  1. 为什么会崩溃?当弹出对象被删除时,创建的 QThread 也会被删除,但 Qt 访问不再分配的内存(核心转储),导致应用程序关闭而不抛出任何异常异常(exception)。

  2. 为什么它在调试期间起作用?许多 IDE(例如 PyCharm)不处理 Qt 错误,因此恕我直言,建议当它们出现此类错误时,在终端/CMD 中执行代码,例如例如,当我执行你的代码时,我获得了:

    QThread: Destroyed while thread is still running
    Aborted (core dumped)
    
  3. 我应该放弃并在应用程序的主窗口中实现进度条(锚定)吗?不。

  4. 我过去已经实现了类似的事情,但没有线程:在工作函数(即 CPU 消耗函数)的循环中,我必须添加 QApplication.processEvents() ,以便在每次迭代时进度条已有效更新。以这种方式做事显然不是最理想的。它仍然是我现在想要实现的更好的替代方案吗? 如果有更好的替代方案,请不要使用 QApplication::processEvents(),在这种情况下,线程是最好的,因为它使主线程不那么繁忙.

<小时/>

最后,初学者在 Qt 中报告的许多错误都涉及变量的范围,因此我建议您分析每个变量应该有多少,例如,如果您希望一个对象与然后类使该变量成为类的属性,如果您只在方法中使用它,那么它只是一个局部变量,等等。

关于python - PyQt5:使用QThread弹出进度条,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59959562/

相关文章:

python - 连接字符串列表中的选定字符串

python - 将装饰器与 Python Slack API 一起使用

python - 未加权双向图上的广度优先搜索

python - 如何平均分割水平布局或指定尺寸

python - 提供用于批处理的输入和输出文件?

python - QApplication.setOverrideCursor对该函数没有任何作用,为什么?

python-3.x - PyQt 或 PySide : QTextEdit deselect all

python - 我想要三个已安装的 PyQt DLL 目录中的哪一个?

python - 在运行 scrapy 之前插入多个输入字段

python - 在MatplotlibWidget pyqt5中使用FigureCanvasQTAgg时出错