从 PyQt5 窗口闪退问题看 Python 垃圾回收与消息机制

前言

此篇文章源于知乎上的一个问题,使用 PyQt5 编写 GUI 程序时,新创建的界面会闪退,本篇文章仅作记录以防以后忘记。

问题代码

import sysfrom PyQt5.QtWidgets import QApplication, QMainWindow, QPushButtonclass Main(QMainWindow):def __init__(self):super(Main, self).__init__()self.button = QPushButton('test', self)self.button.clicked.connect(self.start)self.show()def start(self):w = QMainWindow()w.show()if __name__ == "__main__":app = QApplication(sys.argv)w1 = Main()sys.exit(app.exec_())

在运行以上代码时,会显示一个简单的界面,界面上有一个按钮,如下图:

在这里插入图片描述

当点击 test 按钮时,本应该弹出一个新的窗口,但实际情况却是,有一个窗口一闪而过,重复点击新的窗口仍然一闪而过,快到我根本无法截到窗口打开的样子。

看到这里,大家可以猜测下为什么会出现这种情况,可以从变量作用域方面来猜。

问题原因

下面公布答案,关于窗口闪退的原因其实很简单,那就是 w 变量被回收了。

w 变量的作用范围只在 start 函数内,start 函数运行结束后函数内的所有变量都会被垃圾回收掉,垃圾回收时会自动执行对象中对应的 __del__ 函数(又被叫做析构函数),之后对象被销毁,像没来过一样。

了解了以上知识后来解释窗口闪退的原因就很方便了,w 对象是一个窗口对象,它被创建后紧接着调用了 show 方法,show 方法会将窗口显示在屏幕上,之后 start 函数执行完毕,解释器要清理掉函数作用域内所有的局部变量,自动执行它们的 __del__ 函数,这时窗口就会被回收掉,所以窗口就会自动关闭了。

由于处理器执行的速度非常快,我上面描述的流程在一瞬间就执行结束了,所以在我们看来窗口就像是”闪退“ 了,实际上它只是存在的时间太短了而已。

让我们添加一个延时函数再来看看结果:

在这里插入图片描述

import sys
import timefrom PyQt5.QtWidgets import QApplication, QMainWindow, QPushButtonclass Main(QMainWindow):def __init__(self):super(Main, self).__init__()self.button = QPushButton('test', self)self.button.clicked.connect(self.start)self.show()def start(self):w = QMainWindow()w.show()time.sleep(3)print('start end!')if __name__ == "__main__":app = QApplication(sys.argv)w1 = Main()sys.exit(app.exec_())

执行以上代码,会发现可以正常打开第二个窗口了,但是在三秒过后窗口仍然自己关闭了,那是因为延时时间到了解释器开始了清理工作,窗口对象就被清理掉了。

而且在这三秒钟内我们会发现窗口不响应我们对窗口发出的任何动作,包括最大化、最小化和关闭按钮统统无效,这是 Windows 消息机制的原因。test 按钮绑定了事件处理函数 start,在点击 test 按钮后,按钮单击消息会被发出,放到消息队列中, 消息处理函数依次处理消息队列中的消息,当处理到 start 函数时,会遇到 sleep 函数,此时消息处理函数会等待,等待的过程中也就无法处理其他的消息,导致界面无响应,也就无法处理我们发出的任何消息(消息会被阻塞住,直到消息处理函数恢复正常)。

注:PyQt5 中有一个新的概念,将消息机制抽象为信号和槽,通过信号和槽绑定的方式来传递信号(消息),这样一来方便了后台线程和前台界面进行交互,这种抽象其实大大简化了我们的理解难度。

问题解决

理解了原理后修复这个问题就变得很容易了,既然函数结束变量会被回收,那么将窗口对象绑定到 self 上就可以了,self 会在整个程序退出时才会被销毁,这样就可以防止新打开的窗口闪退了。

import sysfrom PyQt5.QtWidgets import QApplication, QMainWindow, QPushButtonclass Main(QMainWindow):def __init__(self):super(Main, self).__init__()self.button = QPushButton('test', self)self.button.clicked.connect(self.start)self.show()self.w = Nonedef start(self):self.w = QMainWindow()self.w.show()if __name__ == "__main__":app = QApplication(sys.argv)w1 = Main()sys.exit(app.exec_())

问题验证

问题已经解决了,下面来验证下 start 函数执行完毕解释器是怎么销毁变量的,根据上面的结论我们知道,解释器在垃圾回收时会调用对象对应的 __del__ 函数,那么我们重写 QMainWindow 类的 __del__ 函数,观察是否会调用到我们重写的 __del__ 函数。

import sysfrom PyQt5.QtWidgets import QApplication, QMainWindow, QPushButtonclass M(QMainWindow):def __del__(self):print('call __del__')class Main(QMainWindow):def __init__(self):super(Main, self).__init__()self.button = QPushButton('test', self)self.button.clicked.connect(self.start)self.show()self.w = Nonedef start(self):w = M()w.show()if __name__ == "__main__":app = QApplication(sys.argv)w1 = Main()sys.exit(app.exec_())

运行以上代码,我们发现,窗口闪退时,call __del__ 被打印出来了,解释器调用了我们重写后的 __del__ 函数,说明窗口对象的确是被 Python 解释器销毁了。

总结

在本次解决问题的过程中,我们可以学到 __del__ 函数的用法,还了解了 Windows 的消息机制,说明多踩坑还是可以学到不少东西的,哈哈哈🤣

参考链接

https://docs.python.org/zh-cn/3.8/reference/datamodel.html?highlight=del#object.del

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/8215.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

java两个线程的通信/指令重排

目录 1.线程通信 2.指令重排 1.线程通信 前段时间面试笔试题,手写两个线程之间的通信。 问题回顾: 一个生产者,一个消费者,一个桌子媒介。两个线程分别是消费者和生产者。流程是生产者生产一个件商品,通知消费者消…

建立用邻接矩阵表示的无向图

建立用邻接矩阵表示的无向图 #include<stdio.h> #define NUM 100 typedef struct {char vexs[NUM];int edges[NUM][NUM];int n,e; }MGraph; void CreateMGraph(MGraph *G) {int i,j,k;printf("请输入顶点数和边数:");scanf("%d,%d",&G->n,&a…

大模型微调技术 --> LoRA 系列之 AdaLoRA

AdaLoRA 1.摘要 之前的微调方法(如低秩更新)通常将增量更新的预算均匀地分布在所有预训练的权重矩阵上&#xff0c;并且忽略了不同权重参数的不同重要性。结果&#xff0c;微调结果不是最优的。 为了弥补这一差距&#xff0c;我们提出了AdaLoRA&#xff0c;它根据权重矩阵的…

革新网络管理:拉线网套技术引领行业新趋势

根据QYResearch调研团队的最新力作《全球拉线网套市场报告2023-2029》预测&#xff0c;到2029年&#xff0c;全球拉线网套市场的规模有望达到11.7亿美元&#xff0c;且在未来几年内&#xff0c;将以3.5%的复合年增长率&#xff08;CAGR&#xff09;持续扩张。 在全球范围内&…

HTB:Devel[WriteUP]

目录 连接至HTB服务器并启动靶机 1.What is the name of the service is running on TCP port 21 on the target machine? 使用nmap对靶机TCP端口进行开放扫描 2.Which basic FTP command can be used to upload a single file onto the server? 尝试匿名连接至靶机FTP服…

Hadoop集群的高可用(HA)-(2、搭建resourcemanager的高可用)

第一步&#xff1a;检查mapred-site.xml &#xff0c;里面只有yarn配置和historyServer的配置&#xff0c;不需要修改 第二步&#xff1a;修改yarn-site.xml <?xml version"1.0"?> <!-- Licensed under the Apache License, Version 2.0 (the "Lic…

golang笔记

golang笔记 一、内存逃逸 本应在栈中内存,被分配到了堆中 1 返回指针对象 在外部被使用 2 reutrn 函数 使用了上面方法的敞亮 3 入参是interface{} 动态参数 4 make超过栈大小 -gcflags"-m"查看分配内存信息 返回变量vs返回指针 返回变量, 会多一步复制变量, 返回…

纹理分析——统计分析方法

一. 灰度共生矩阵法(Gray Level Co-occurrence Matrix, GLCM ) 灰度共生矩阵又称为灰度空间相关矩阵&#xff0c;是通过研究灰度的空间相关特性来描述纹理的常用方法。&#xff08;也称为联合概率矩阵&#xff09;它作为传统的图像纹理分析方法已广泛应用于数字图像处理的许多…

IT维修记录表导入接口的思路

上篇文章讲了IT设备信息表的导入接口的思路&#xff0c;这篇文章趁热打铁&#xff0c;把IT维修记录表的导入接口的思路给说一下。 首先我们要知道IT维修记录表的数据是什么来的&#xff1f;这个问题必须要搞懂&#xff0c;不搞懂的话对接下来的思路其实是不利的。IT维修记录表…

场景解决方案丨迎战电商大促,企业管理跟踪驾驶舱助力中小企业打赢决胜之战

该方案已沉淀为➡️订单物流信息跟踪模板&#xff0c;点击&#x1f517;即可体验 随着互联网技术的发展和市场经济的变化&#xff0c;各行业的线上竞争愈发激烈。一方面&#xff0c;互联网平台凭借便捷的服务和丰富的产品吸引了大量客户&#xff1b;另一方面&#xff0c;复杂多…

WebRTC 环境搭建

主题 本文主要描述webrtc开发过程中所需的环境搭建 环境&#xff1a; 运行环境&#xff1a;ubuntu 20.04 Node.js环境搭建 安装编译 Node.js 所需的依赖包: sudo apt-get update sudo apt-get install -y build-essential libssl-dev 下载 Node.js 源码: curl -sL htt…

Python从入门到高手7.5节-实现冒泡排序算法

目录 7.5.1 排序算法简介 7.5.2 冒泡排序算法原理 7.5.3 冒泡排序算法实现 7.5.4 永不放弃 7.5.1 排序算法简介 所谓排序&#xff0c;是指将数据集合中的元素按从小到大的顺序进行排列&#xff0c;或按从大到小的顺序进行排列。 前者称为升序排序&#xff0c;后者称为降序…

vue-quill-editor富文本编辑器

效果图&#xff1a; 1、下载安装vue-quill-editor npm install vue-quill-editor --save图片缩放、拖拽 npm install quill-image-drop-module -S //允许粘贴图像并将其拖放到编辑器中。 npm install quill-image-resize-module -S //允许调整图像大小<template>&…

TCP是怎样工作的网络拥塞控制理论和算法部分记录

参考资料 https://github.com/ituring/tcp-book 流量控制、窗口控制和拥塞控制的关系 流量控制、窗口控制和拥塞控制的关系如图所示 窗口控制是上层的概念&#xff0c;核心思路是基于滑动窗口技术传输数据。而确定发送窗口大小的方法有流量控制和拥塞控制两种 流量控制&…

NVR管理平台EasyNVR多个NVR同时管理对接天翼云云存储的一些关键信息和优势

在视频监控领域&#xff0c;随着技术的不断进步&#xff0c;存储方式的选择变得尤为重要。传统的本地存储方式受限于硬件容量&#xff0c;而云存储则以其强大的数据处理能力和弹性扩展性&#xff0c;成为视频数据存储的理想选择。NVR管理平台EasyNVR作为一款领先的视频汇聚与管…

饲料加工机器设备由搅拌机粉碎机颗粒机组成

饲料加工机器设备在现代养殖业中扮演着至关重要的角色&#xff0c;它们不仅提高了饲料的生产效率&#xff0c;还优化了饲料的营养价值。其中&#xff0c;饲料粉碎机、搅拌机和颗粒机是饲料加工流程中的三大核心设备。 想象一下&#xff0c;一把把粗糙的原料&#xff0c;在粉碎…

oracle数据坏块处理(二)-逻辑坏块重新格式化处理

1、问题描述 在使用duplicate搭建DG时报错 包括rman copy&#xff0c;rman备份 2、问题分析 由于数据文件逻辑坏块导致物理备份不能正常进行。 使用rman检查数据文件47 SELECT tablespace_name, segment_type, owner, segment_name FROM dba_extents WHERE file_id 47 a…

在IDEA使用arthas实现jar包方法耗时统计

1.背景 对于依赖jar包中的方法内部耗时统计&#xff0c;传统的手写StopWatch不适用&#xff0c;这儿采用arthas统计。 官网文档比较详细&#xff0c;trace | arthas 使用版本&#xff1a; arthas-boot version: 4.0.2 IntelliJ IDEA 2023.3.3 2.使用介绍 2.1.启动需要检…

用于图像识别的判别图正则化技术

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f916;编程探索专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2024年11月8日13点32分 点击开启你的论文编程之旅https://www.aspiringcode.com/content?id17210272021224&uid64a84f9640714755a…

Android Handler

Handler用于多线程消息分发和处理。与handler相关的几个对象&#xff1a;Message, Looper&#xff0c;MessageQueue, ThreadLocal. Handler是Message的消费者。 MessageQueue是容器。 Looper是整个Message分发的驱动。 Handler中有多种发送消息的方法&#xff0c;其中postxx…