• 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_())
    View Code

    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_())
    View Code

    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_())
    View Code

    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_())
    复制代码

     

  • 相关阅读:
    基于springboot的ShardingSphere5.2.1的分库分表的解决方案之mysql主从的配置(五)
    序列化(二)Parcelable
    git clone项目报错文件名过长
    js的构造函数
    做个小工具显示UE里地形的高度图和权重图
    MapReduce论文解读
    ES6 入门教程 17 Promise 对象 17.1 Promise 的含义
    mqtt使用方法
    【通信、算法、旅游、人工智能、图像处理、机械、医疗】EI会议(2023)
    ContentProvider的执行时机
  • 原文地址:https://www.cnblogs.com/lihongtaoya/p/18197784