【问题标题】:msgbox error while threading , GUI blocks [closed]线程时出现msgbox错误,GUI块[关闭]
【发布时间】:2021-02-22 09:59:36
【问题描述】:

我在执行以下 gui 时遇到问题。如果没有消息框,它可以正常工作,但是当有消息框时,它会阻塞。知道为什么有消息时gui会阻塞。谢谢

from PyQt5 import QtCore, QtGui, QtWidgets
import threading
import time
class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(800, 600)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.verticalLayout = QtWidgets.QVBoxLayout(self.centralwidget)
        self.verticalLayout.setObjectName("verticalLayout")
        self.pushButton = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton.setObjectName("pushButton")
        self.verticalLayout.addWidget(self.pushButton)
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 800, 26))
        self.menubar.setObjectName("menubar")
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(MainWindow)
        self.statusbar.setObjectName("statusbar")
        MainWindow.setStatusBar(self.statusbar)
        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)
        self.pushButton.pressed.connect(self.threadingc)      
    def calculation(self):
        for i in range(10):
            time.sleep(1)
            print(i)
        msg = QtWidgets.QMessageBox()
        msg.setInformativeText('Finish')
        msg.exec_()   
    def threadingc(self):
        x=threading.Thread(target=self.calculation)
        x.start()
    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
        self.pushButton.setText(_translate("MainWindow", "PushButton"))
import sys
if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_())

【问题讨论】:

  • 请避免创建与您发布的其他问题非常相似的问题。回答这个问题后,我才注意到您的other post。我将把它留在这里,因为我相信它回答了你的问题,但我建议你删除另一个,因为它也可以通过这个来回答(并且在 StackOverflow 上已经有很多类似的帖子)。

标签: python multithreading pyqt5 qthread


【解决方案1】:

任何对 UI 元素的访问只允许在主 Qt 线程内进行。请注意,access 不仅意味着读取/写入小部件属性,还意味着创建;任何从其他线程执行此操作的尝试在最好的情况下都会导致图形问题或不一致的行为,在最坏(和更常见)的情况下会导致崩溃。

这样做的唯一正确方法是使用带有(可能)自定义信号的 QThread:这允许 Qt 正确地对信号进行排队,并在它可以实际处理它们时对它们做出反应。

以下是一个非常简单的情况,不需要创建 QThread 子类,但请考虑这只是为了教育目的。

class Ui_MainWindow(object):
    # ...

    def calculation(self):
        for i in range(10):
            time.sleep(1)
            print(i)

    def showMessage(self):
        msg = QtWidgets.QMessageBox()
        msg.setInformativeText('Finish')
        msg.exec_()   
        self.pushButton.setEnabled(True)

    def threadingc(self):
        self.pushButton.setEnabled(False)
        self.thread = QtCore.QThread()
        # override the `run` function with ours; this ensures that the function
        # will be executed in the new thread
        self.thread.run = self.calculation
        self.thread.finished.connect(self.showMessage)
        self.thread.start()

请考虑以下重要方面:

  • 我必须禁用按钮,否则可能会在前一个线程仍在执行时创建一个新线程;这会产生问题,因为覆盖 self.thread 会导致 python 在运行时尝试对前一个线程进行垃圾收集(删除),这是一件非常糟糕的事情;
  • 一个可能的解决方案是创建具有父级的线程,这通常使用简单的QThread(self) 完成,但在您的情况下这是不可能的,因为 Qt 对象只能接受其他 Qt 对象作为其父级,而在你的案例self 将是一个Ui_MainWindow 实例(这是一个基本的python 对象);
  • 以上几点是一个重要问题,因为您正试图从pyuic 生成的文件开始执行您的程序,这应该永远完成:这些文件打算保留为它们没有任何手动修改,仅用作导入模块;在有关using Designer 的官方指南中阅读有关此主题的更多信息;另请注意,试图模仿这些文件的行为是没有用的,因为这通常会导致对象结构非常混乱;
  • 理论上,您可以添加对 qt 对象的引用(例如,通过在 setupUi() 函数中添加 self.mainWindow = MainWindow)并使用该引用 (thread = QThread(self.mainWindow)) 创建线程,或者将线程添加到持久列表(self.threads = [],再次在 setupUi() 中),但由于上述观点,我强烈建议您不要这样做;

最后,更正确代码的实现将要求您再次生成 ui 文件,保持原样并执行类似以下示例的操作;请注意,我添加了一个非常基本的异常实现,它还展示了如何正确地与自定义信号交互。

from PyQt5 import QtCore, QtGui, QtWidgets
from mainwindow import Ui_MainWindow
import time

class Calculation(QtCore.QThread):
    error = QtCore.pyqtSignal(object)
    def run(self):
        for i in range(10):
            time.sleep(1)
            print(i)
            try:
                10 / 0
            except Exception as e:
                self.error.emit(e)
                break


class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self):
        super().__init__()
        self.setupUi(self)
        self.pushButton.pressed.connect(self.threadingc)

    def showMessage(self):
        msg = QtWidgets.QMessageBox()
        msg.setInformativeText('Finish')
        msg.exec_()   

    def threadingc(self):
        # create the thread with the main window as a parent, this is possible 
        # since QMainWindow also inherits from QObject, and this also ensures
        # that python will not delete it if you want to start another thread
        thread = Calculation(self)
        thread.finished.connect(self.showMessage)
        thread.error.connect(self.showError)
        thread.start()


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    mainWindow = MainWindow()
    mainWindow.show()
    sys.exit(app.exec_())

在上述情况下,使用以下命令处理 ui 文件(显然,假设 ui 名为“mainwindow.ui”):

pyuic mainwindow.ui -o mainwindow.py

【讨论】:

  • 感谢您的回复,这很有趣,我学到了很多。在calculation.run中想象一下我有一个异常或很多异常,在那个异常中我想看到一个msgbox我该怎么做。谢谢
  • @Yakoub 查看更新
  • 这会很有帮助,谢谢
猜你喜欢
  • 2013-03-14
  • 2019-03-01
  • 2021-10-14
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-07-18
  • 2019-03-27
  • 2013-07-18
相关资源
最近更新 更多