一.简介

在使用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的更多相关文章

  1. Java子线程中操作主线程Private级别数据

    两个类分别如下: <pre name="code" class="java">package Demo2; import java.util.*; ...

  2. Android——子线程操作主线程

    子线程不能直接操作主线程 UI线程 //水平进度条 public void jdt1_onclick(View view) { final ProgressDialog pd = new Progre ...

  3. 在C#中子线程如何操作主窗口线程上的控件

    在C#中子线程怎样操作主线程中窗口上控件 在C#中,直接在子线程中对窗口上的控件操作是会出现异常,这是因为子线程和运行窗口的线程是不同的空间,因此想要在子线程来操作窗口上的控件.是不可能简单的通过控件 ...

  4. MFC线程内操作主窗体 控件

    CWnd* h_d2 = AfxGetApp()->GetMainWnd(); //获取主窗口的句柄 h_d2-> GetDlgItem(IDC_EDIT2)->GetWindowT ...

  5. Android中进程与线程及如何在子线程中操作UI线程

    1. Android进程 一个应用程序被启动时,系统默认创建执行一个叫做"main"的线程.这个线程也是你的应用与界面工具包(android.widget和android.view ...

  6. Java多线程开发系列之三:线程这一辈子(线程的生命周期)

    前文中已经提到了,关于多线程的基础知识和多线程的创建.但是如果想要很好的管理多线程,一定要对线程的生命周期有一个整体概念.本节即对线程的一生进行介绍,让大家对线程的各个时段的状态有一定了解. 线程的一 ...

  7. Java多线程系列--“基础篇”10之 线程优先级和守护线程

    概要 本章,会对守护线程和线程优先级进行介绍.涉及到的内容包括:1. 线程优先级的介绍2. 线程优先级的示例3. 守护线程的示例 转载请注明出处:http://www.cnblogs.com/skyw ...

  8. eventloop & actor模式 & Java线程模型演进 & Netty线程模型 总结

    eventloop的基本概念可以参考:http://www.ruanyifeng.com/blog/2013/10/event_loop.html Eventloop指的是独立于主线程的一条线程,专门 ...

  9. Java中的守护线程和非守护线程(转载)

    <什么是守护线程,什么是非守护线程> Java有两种Thread:"守护线程Daemon"(守护线程)与"用户线程User"(非守护线程). 用户线 ...

  10. 【温故而知新-万花筒】C# 异步编程 逆变 协变 委托 事件 事件参数 迭代 线程、多线程、线程池、后台线程

    额基本脱离了2.0 3.5的时代了.在.net 4.0+ 时代.一切都是辣么简单! 参考文档: http://www.cnblogs.com/linzheng/archive/2012/04/11/2 ...

随机推荐

  1. 体验Semantic Kernel图片内容识别

    前言 前几日在浏览devblogs.microsoft.com的时候,看到了一篇名为Image to Text with Semantic Kernel and HuggingFace的文章.这篇文章 ...

  2. [原创工具] 病毒整理器V0.98

    最近由于各种需要,所以开发了一个. RT,下载地址: 链接:https://pan.baidu.com/s/1Kxd77-n3fbPZQm_CIH6LTA 提取码:xq8f 很简单的一个小工具,以后还 ...

  3. HMS Core机器学习服务,高效助力跨语种沟通

    5月24日, HUAWEI Developer Day(简称HDD)线上沙龙·创新开发专场活动成功举办.HMS Core机器学习服务(ML Kit)产品经理在会上围绕机器翻译的技术优势.使用场景和接入 ...

  4. Python格式化字符串:%、format、f-string

    目前Python格式化字符串的方式有三种: 1. % 2.format 3.f-string % 格式化常用方法: # % 格式化字符串 s1 = 'name is %s' % ('zhangsan' ...

  5. shell编程实现用户循环输入

    如果你想在Shell脚本中实现一个循环判断用户输入是否正确,并根据情况决定是否退出系统,可以使用一个无限循环(如while true)和条件语句来实现. 以下是一个示例代码,用于演示这种情况: #!/ ...

  6. RC4Drop加密技术:原理、实践与安全性探究

    第一章:介绍 1.1 加密技术的重要性 加密技术在当今信息社会中扮演着至关重要的角色.通过加密,我们可以保护敏感信息的机密性,防止信息被未经授权的用户访问.窃取或篡改.加密技术还可以确保数据在传输过程 ...

  7. 【高级RAG技巧】在大模型知识库问答中增强文档分割与表格提取

    前言 文档分割是一项具有挑战性的任务,它是任何知识库问答系统的基础.高质量的文档分割结果对于显著提升问答效果至关重要,但是目前大多数开源库的处理能力有限. 这些开源的库或者方法缺点大致可以罗列如下: ...

  8. 《C# in depth》第5章C#5.0中的更改(十三)——異步枚舉器

    一.異步枚舉 异步枚举器(Async Enumerator)是指一种异步迭代器,可以用于处理异步数据源.它允许我们以异步的方式逐个读取数据源中的元素. 在传统的同步枚举器中,当我们遍历一个集合时,程序 ...

  9. 力扣278(java&python)-第一个错误的版本(简单)

    题目: 你是产品经理,目前正在带领一个团队开发新的产品.不幸的是,你的产品的最新版本没有通过质量检测.由于每个版本都是基于之前的版本开发的,所以错误的版本之后的所有版本都是错的. 假设你有 n 个版本 ...

  10. 第3章 python 爬虫抓包与数据解析

    第 3章 Python 爬虫抓包与数据解析 3.1 抓包进阶 目前,我们已经会使用 Chrome 浏览器自带的开发者工具来抓取访问网页的数据包,但是这种抓包方法有局限性,比如只能监听一个浏览器选项卡, ...