PyQt5:使用QThread的弹出进度栏

问题描述 投票:0回答:1

我如何在弹出窗口中实现进度条,该方法通过以下方法监视所谓的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社区2019.3中,该程序崩溃并且我没有得到任何清晰的错误消息。

不过,当我调试它时,它看起来像是可以正常工作的,因为我能够看到要实现的目标:App working during debugging

我有一系列问题:

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

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

python pyqt pyqt5 qthread qprogressbar
1个回答
0
投票

说明:

要了解该问题,您必须了解以下内容:

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")
self.popup = PopUpProgressB()
self.main_window_button.clicked.connect(self.popup.show)
self.h_box.addWidget(self.main_window_button)
# ...

因此,它不会显示,您必须删除对PopUpProgressB的show()方法的调用:

class PopUpProgressB(QWidget):
    def __init__(self):
        super().__init__()
        # ...
        self.setWindowTitle('Progress Bar')
        # self.show() # 
        self.obj = Worker()
        # ...

由于我已经解释了您的问题的失败,所以我会回答您的问题:

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

    ] >
  2. [为什么在调试过程中起作用?

  3. 许多IDE(例如PyCharm)不处理Qt错误,因此恕我直言,建议当它们出现此类错误时,它们在终端/ CMD中执行其代码,例如,当我执行我获得的代码:
    QThread: Destroyed while thread is still running
    Aborted (core dumped)
    
  4. 我应该放弃并在应用程序的主窗口中实现进度栏(锚定)吗?

  5. 编号
  6. 我过去已经实现了类似的操作,但是没有线程:在worker函数(即消耗CPU的函数)的循环中,我必须添加QApplication.processEvents(),以便在每次迭代时都可以看到进度条有效更新。用这种方式做事显然不是最理想的。 [

  7. 如果有更好的选择,请不要使用QApplication :: processEvents(),在这种情况下,线程是最好的,因为它使主线程不那么忙碌。] >

    最后,初学者在Qt中报告的许多错误都涉及变量的范围,因此,我建议您分析每个变量应包含的数量,例如,如果您希望对象与类保持相同的状态然后使该变量成为类的属性,如果相反,您仅在方法中使用它,则它仅是局部变量,等等。

© www.soinside.com 2019 - 2024. All rights reserved.