python - 线程时出现 msgbox 错误,GUI 块
问题描述
执行以下 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_())
解决方案
仅允许从主 Qt 线程内访问 UI 元素。请注意,访问不仅意味着读取/写入小部件属性,还意味着创建;任何从其他线程执行此操作的尝试在最好的情况下都会导致图形问题或不一致的行为,在最坏(和更常见)的情况下会导致崩溃。
这样做的唯一正确方法是使用带有(可能)自定义信号的 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)
. (这是一个基本的python对象);self
Ui_MainWindow
- 以上几点是一个重要问题,因为您试图从
pyuic
生成的文件开始执行程序,而这绝不应该这样做:这些文件旨在保持原样而无需任何手动修改,并且仅用作导入的模块; 在有关使用 Designer的官方指南中阅读有关此主题的更多信息;还要注意,试图模仿这些文件的行为是没有用的,因为这通常会导致对对象结构的极大混淆; - 您理论上可以添加对 qt 对象的引用(例如,通过添加函数)并使用该引用创建线程(
self.mainWindow = MainWindow
),或者将线程添加到持久列表(再次在 中),但由于上述原因我强烈不鼓励你这样做;setupUi()
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
推荐阅读
- django - AWS Elastic Beanstalk 日志时间部署 Python Django
- database - 我可以使用 jaguardb 来处理流数据吗?
- pandas - 如何将函数的返回值添加到数据框列中?
- windows - 获取 Windows 用户帐户(配置文件)列表
- django - 如何设置我的 crontab 以便在 Django 中执行 process_tasks?
- php - 如何突出显示要下载的文件中的字符串?
- html - 指针事件 - 当我使用制表键时没有一个不起作用
- c# - 什么检查方法比设置一个值是最优化的?
- kubernetes - Kubernetes “不错” pod 的 CPU 使用率
- objective-c - 目标 c 在完成处理程序上得到 EXC_BAD_ACCESS 错误