灌篮少年 发表于 2024-5-20 10:51:38

pyqt5 子线程如何操作主线程GUI

一.简介

在使用pyqt5编写gui时碰到两个问题,会导致界面崩溃,今天就围绕这两个问题来简单说明和改进。
1.在主线程中使用while无限循环会导致界面崩溃
2.在子线程中操作主线程gui会导致界面崩溃
二.步骤说明

1.在主线程中使用while无限循环会导致界面崩溃

1)错误代码
https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gifhttps://images.cnblogs.com/OutliningIndicators/ExpandedBlockStart.gifimport sys

from PyQt5.QtWidgets importQPushButton, 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 Code2)崩溃原因
我们先来说下while崩溃的问题,这边我设置的循环是一个无限循环,不会给 GUI 变乱循环任何运行的时机。在 PyQt 或其他 GUI 框架中,GUI 的变乱循环(例如按钮点击、窗口移动等)必须在单独的线程中运行,以保持 GUI 的响应性
3)改进方法
将循环体在一个子线程中执行,就可以避免这个问题,代码如下。
https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gifhttps://images.cnblogs.com/OutliningIndicators/ExpandedBlockStart.gifimport 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 Code2.在子线程中操作主线程gui会导致界面崩溃

1)错误代码
https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gifhttps://images.cnblogs.com/OutliningIndicators/ExpandedBlockStart.gifimport 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 Code2)崩溃原因
上述中试图在子线程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 sysimport threadingimport timefrom PyQt5.QtCore import QObject, pyqtSignalfrom PyQt5.QtWidgets import QPushButton, QTextEdit, QApplication, QHBoxLayout, QWidgetclass 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_()) 

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: pyqt5 子线程如何操作主线程GUI