pyqt5 子线程如何操作主线程GUI(示例代码)
一.简介
在使用pyqt5编写gui时遇到两个问题,会导致界面崩溃,今天就围绕这两个问题来简单说明和改进。
1.在主线程中使用while无限循环会导致界面崩溃
2.在子线程中操作主线程gui会导致界面崩溃
二.步骤说明
1.在主线程中使用while无限循环会导致界面崩溃
1)错误代码
import sys from PyQt5.QtWidgets import QPushButton, QTextEdit, QApplication, QHBoxLayout, QWidget class FileChooserApp(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): button = QPushButton("按钮") self.reviewEdit = QTextEdit() self.reviewEdit.move(100, 100) button.clicked.connect(self.send) hbox1 = QHBoxLayout() # 创建一个水平布局 hbox1.addWidget(button) # 添加按钮到水平布局中 hbox1.addStretch(1) # 设置水平比例间距 hbox1.addWidget(self.reviewEdit) # 添加按钮到水平布局中 self.setLayout(hbox1) # 添加到布局器 self.setWindowTitle('文件选择器') self.setGeometry(300, 300, 500, 500) def send(self): """ 事件 :return: """ while True: """ 逻辑代码 """ self.reviewEdit.setText("测试") if __name__ == '__main__': app = QApplication(sys.argv) ex = FileChooserApp() ex.show() sys.exit(app.exec_())
2)崩溃原因
我们先来说下while崩溃的问题,这边我设置的循环是一个无限循环,不会给 GUI 事件循环任何运行的机会。在 PyQt 或其他 GUI 框架中,GUI 的事件循环(例如按钮点击、窗口移动等)必须在单独的线程中运行,以保持 GUI 的响应性
3)改进方法
将循环体在一个子线程中执行,就可以避免这个问题,代码如下。
import sys import threading from PyQt5.QtWidgets import QPushButton, QTextEdit, QApplication, QHBoxLayout, QWidget class FileChooserApp(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): button = QPushButton("按钮") self.reviewEdit = QTextEdit() self.reviewEdit.move(100, 100) button.clicked.connect(self.send) hbox1 = QHBoxLayout() # 创建一个水平布局 hbox1.addWidget(button) # 添加按钮到水平布局中 hbox1.addStretch(1) # 设置水平比例间距 hbox1.addWidget(self.reviewEdit) # 添加按钮到水平布局中 self.setLayout(hbox1) # 添加到布局器 self.setWindowTitle('文件选择器') self.setGeometry(300, 300, 500, 500) def send(self): """ 事件 :return: """ def send_a(): while True: """ 逻辑代码 """ print("123") send_thread = threading.Thread(target=send_a) # 启动线程 send_thread.start() if __name__ == '__main__': app = QApplication(sys.argv) ex = FileChooserApp() ex.show() sys.exit(app.exec_())
2.在子线程中操作主线程gui会导致界面崩溃
1)错误代码
import sys import threading import time from PyQt5.QtWidgets import QPushButton, QTextEdit, QApplication, QHBoxLayout, QWidget class FileChooserApp(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): button = QPushButton("按钮") self.reviewEdit = QTextEdit() self.reviewEdit.move(100, 100) button.clicked.connect(self.send) hbox1 = QHBoxLayout() # 创建一个水平布局 hbox1.addWidget(button) # 添加按钮到水平布局中 hbox1.addStretch(1) # 设置水平比例间距 hbox1.addWidget(self.reviewEdit) # 添加按钮到水平布局中 self.setLayout(hbox1) # 添加到布局器 self.setWindowTitle('文件选择器') self.setGeometry(300, 300, 500, 500) def send(self): """ 事件 :return: """ def send_a(): while True: """ 逻辑代码 """ self.reviewEdit.setText("设置文案") send_thread = threading.Thread(target=send_a) # 启动线程 send_thread.start() if __name__ == '__main__': app = QApplication(sys.argv) ex = FileChooserApp() ex.show() sys.exit(app.exec_())
2)崩溃原因
上述中试图在子线程send_a方法中给文本编辑器设置文案。这是不允许的,因为 PyQt 和大多数 GUI 框架一样,要求所有的 GUI 更新必须在主线程(也称为 GUI 线程)中执行。
3)改进方法
既然在子线程中无法操作主线程gui更新,那么只能在主线程中去执行,这就需要信号与槽的配合了。我们先来自定义一个信号
class YourThread(QObject): show_warning_signal = pyqtSignal() def run(self): self.show_warning_signal.emit()
在初始化的时候去实例化YourThread类,连线信号与槽
class FileChooserApp(QMainWindow): def __init__(self): super().__init__() self.initUI() self.your = YourThread() self.your.show_warning_signal.connect(self.settext)
接着在子线程中直接去触发信号即可
def send(self): def send_a(): while True: """ 循环体 """ self.your.run() time.sleep(2) send_thread = threading.Thread(target=send_a) # 启动线程 send_thread.start()
执行每次循环需要等待2s,避免出现无限循环导致应用程序冻结、响应缓慢或其他线程相关的问题
三.实例
import sys import threading import time from PyQt5.QtCore import QObject, pyqtSignal from PyQt5.QtWidgets import QPushButton, QTextEdit, QApplication, QHBoxLayout, QWidget class YourThread(QObject): show_warning_signal = pyqtSignal() def run(self): self.show_warning_signal.emit() class FileChooserApp(QWidget): def __init__(self): super().__init__() self.initUI() self.your = YourThread() self.your.show_warning_signal.connect(self.settext) def initUI(self): button = QPushButton("按钮") self.reviewEdit = QTextEdit() self.reviewEdit.move(100, 100) button.clicked.connect(self.send) hbox1 = QHBoxLayout() # 创建一个水平布局 hbox1.addWidget(button) # 添加按钮到水平布局中 hbox1.addStretch(1) # 设置水平比例间距 hbox1.addWidget(self.reviewEdit) # 添加按钮到水平布局中 self.setLayout(hbox1) # 添加到布局器 self.setWindowTitle('文件选择器') self.setGeometry(300, 300, 500, 500) def send(self): """ 事件 :return: """ def send_a(): while True: """ 逻辑代码 """ self.your.run() time.sleep(2) send_thread = threading.Thread(target=send_a) # 启动线程 send_thread.start() def settext(self): self.reviewEdit.setText("123") if __name__ == '__main__': app = QApplication(sys.argv) ex = FileChooserApp() ex.show() sys.exit(app.exec_())
到此这篇关于pyqt5 子线程如何操作主线程GUI的文章就介绍到这了,更多相关pyqt5 子线程操作主线程GUI内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
使用Python自动化Microsoft Excel和Word的操作方法
这篇文章主要介绍了使用Python自动化Microsoft Excel和Word,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2021-04-04python3.8.1+selenium实现登录滑块验证功能
这篇文章主要介绍了python3.8.1+selenium解决登录滑块验证的问题,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2020-05-05Python之urlencode和urldecode案例讲解
这篇文章主要介绍了Python之urlencode和urldecode案例讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下2021-08-08
最新评论