QT开发(十九)——QT内存泄漏问题

2023-10-27

QT开发(十九)——QT内存泄漏问题

一、QT对象间的父子关系

    QT最基础和核心的类是:QObject,QObject内部有一个list,会保存children,还有一个指针保存parent,当自己析构时,会自己从parent列表中删除并且析构所有的children。

QT对象之间可以存在父子关系,每一个对象都可以保存它所有子对象的指针,每一个对象都有一个指向其父对象的指针。

当指定QT对象的父对象时,父对象会在子对象链表中加入该对象的指针,该对象会保存指向其父对象的指针。

QT对象被销毁时,将自己从父对象的子对象链表中删除,将自己的子对象链表中的所有对象销毁。

QT对象销毁时解除和父对象之间的父子关系,并销毁所有的子对象。

二、QT的半自动化内存管理

    1、QObject及其派生类的对象,如果其parent非0,那么其parent析构时会析构该对象。如果父对象和子对象都分配在栈上,并且先释放父对象的内存空间,释放父对象的时候子对象的空间将会被释放,当释放子对象的空间时,子对象空间已经被释放,会发生内存错误。

    2、QWidget及其派生类的对象,可以设置 Qt::WA_DeleteOnClose 标志位(当close时会析构该对象)。

    3、QAbstractAnimation派生类的对象,可以设置 QAbstractAnimation::DeleteWhenStopped。

    4、QRunnable::setAutoDelete()、MediaSource::setAutoDelete()。

    5、父子关系:父对象、子对象、父子关系。这是Qt中所特有的,与类的继承关系无关,传递参数是与parent有关(基类、派生类,或父类、子类,这是对于派生体系来说的,与parent无关)。

三、QT内存泄漏实例

1、实例一

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QLabel *label = new QLabel("Hello Qt!");

    label->show();

    return a.exec();

}

    label 没有指定parent,也没有对其调用delete,会造成内存泄漏。

    解决方案:

    A、分配对象到栈上而不是堆上

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QLabel label("Hello Qt!");

    label.show();

    return a.exec();

}

    B、设置标志位,close()后会delete label。

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QLabel *label = new QLabel("Hello Qt!");

    label->show();

    label->setAttribute(Qt::WA_DeleteOnClose);

    return a.exec();

}

    C、在堆上new分配空间,delete手动释放

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    int ret = 0;

    QApplication a(argc, argv);

    QLabel *label = new QLabel("Hello Qt!");

    label->show();

    ret = a.exec();

    delete label;

    return ret;

}

2、实例二

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QLabel label("Hello Qt!");

    label.show();

    label.setAttribute(Qt::WA_DeleteOnClose);

    return a.exec();

}

    label对象是在栈上分配的内存空间,delete栈上的地址会出错。

3、实例三

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

  QApplication a(argc, argv);

  QLabel label("Hello Qt!");

    QWidget w;

    label.setParent(&w);

  w.show();

    return a.exec();

}

    w比label先被析构,当w被析构时,会删除chilren列表中的对象label,但label是分配到栈上的,因delete栈上的对象而出错。

    解决方案:

    A、调整父对象的位置,确保父对象析构时子对象已经析构

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QWidget w;

    QLabel label("Hello Qt!");

    label.setParent(&w);

    w.show();

    return a.exec();

}

    B、将子对象分配到堆空间

#include <QApplication>

#include <QLabel>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QLabel *label = new QLabel("Hello Qt!");

    QWidget w;

    label->setParent(&w);

    w.show();

    return a.exec();

}

4、实例四

    当一个QObject正在接受事件队列时中途被销毁了,会出现异常,所以QT中不要直接Delete掉一个QObject,如果一定要做,要使用QObject的deleteLater()函数,deleteLater()函数会让所有事件都发送完一切处理好后清除这片内存,而且就算调用多次的deletelater也不会有问题。

四、智能指针

1、QPointer

    QPointer是一个模板类,QPointer可以监视动态分配空间的对象,并且在对象被 delete 的时候及时更新。

    QPointer的现实原理:在QPointer保存了一个QObject的指针,并把这个指针的指针(双指针)交给全局变量管理,而QObject 在销毁时(析构函数,QWidget是通过自己的析构函数的,而不是依赖QObject的)会调用QObjectPrivate::clearGuards 函数来把全局 GuardHash 的那个双指针置为零,因为是双指针的问题,所以QPointer中指针当然也为零了。用isNull 判断就为空了。

2std::auto_ptr

    auto_ptr被销毁时会自动删除它指向的对象。

五、垃圾回收机制

1QObjectCleanupHandler

    Qt 对象清理器是实现自动垃圾回收的很重要部分。QObjectCleanupHandler可以注册很多子对象,并在自己删除的时候自动删除所有子对象。同时,它也可以识别出是否有子对象被删除,从而将其从它的子对象列表中删除。QObjectCleanupHandler类可以用于不在同层次中的类的清理操作,例如,当按钮按下时需要关闭很多窗口,由于窗口的 parent 属性不可能设置为别的窗口的 button,此时使用QObjectCleanupHandler类就会方便。

#include <QApplication>

#include <QObjectCleanupHandler>

#include <QPushButton>

int main(int argc, char *argv[])

{

    QApplication a(argc, argv);

    QObjectCleanupHandler *cleaner = new QObjectCleanupHandler;

    QPushButton *w = new QPushButton("Remove me");

    w->show();

    cleaner->add(w);

    //点击“remove me”按钮删除自身

    QObject::connect(w, SIGNAL(clicked()), w, SLOT(deleteLater()));

    w = new QPushButton("Nothing");

    cleaner->add(w);

    w->show();

    w = new QPushButton("Remove all");

    cleaner->add(w);

    w->show();

    //点击“remove all”按钮删除所有QObject

    QObject::connect(w, SIGNAL(clicked()), cleaner, SLOT(deleteLater()));

    return a.exec();

}

    点击Remove me”按钮会删除掉自己(通过 deleteLater() 槽),cleaner 会自动将其从自己的列表中清除。点击Remove all”按钮后会删除cleaner,会同时删除掉所有未关闭的窗口,即注册在cleaner的QObject对象


转载于:https://blog.51cto.com/9291927/1871608

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

QT开发(十九)——QT内存泄漏问题 的相关文章

  • netty handler的执行顺序(3)

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 今天解决2个问题 1 handler在pipeline当中究竟是如何存储的 2 在遍历handler的过程中 会根据event的不同 调用不同的handler 这一点是如何
  • python 历险记(五)— python 中的模块

    目录 前言 基础 模块化程序设计 模块化有哪些好处 什么是 python 中的模块 引入模块有几种方式 模块的查找顺序 模块中包含执行语句的情况 用 dir 函数来窥探模块 python 的内置模块有哪些 结语 参考文档 系列文章列表 前言
  • 算法--将数组分成和相等的多个子数组,求子数组的最大个数

    作者 陈太汉 一个整数数组 长度为n 将其分为m份 使各份的和相等 求m的最大值 比如 3 2 4 3 6 可以分成 3 2 4 3 6 m 1 3 6 2 4 3 m 2 3 3 2 4 6 m 3 所以m的最大值为3 算法 原理的思想是
  • 将二叉树转为有序的双向链表

    一 题目要求 输入一棵二叉排序树 现在要将该二叉排序树转换成一个有序的双向链表 而且在转换的过程中 不能创建任何新的结点 只能调整树中的结点指针的指向来实现 include
  • Sort List

    Sort a linked list in O n log n time using constant space complexity 题目要求用 O n log n 的时间复杂度和常数的空间复杂度来进行链表排序 O nlogn 的排序算
  • 直线检测方法—LSD论文翻译

    附原文链接 LSD a Line Segment Detector 摘 要 LSD是一个线段检测器 能够在线性时间内得到亚像素级精度的检测结果 它无需调试参数就可以适用于任何数字图像上 并且能够自我控制错误数量的检测 平均来说 一个图像中允
  • Hash table and application in java

    查找的效率取决于在查找是比较的次数 次数越少效率越高 反之越低 最理想的情况是无需比较 一次存取便能找到所查找的记录 根据对应关系f找到给定值K的像f K hash function 应运而生 由此思想建的表称为hash table 集合h
  • 常用的十种算法--马踏棋盘算法

    1 马踏棋盘算法介绍 马踏棋盘算法也被称为骑士周游问题 将马随机放在国际象棋的 8 8 棋盘 Board 0 7 0 7 的某个方格中 马按走棋规则 马走日字 进行移动 要求每个方格只进入一次 走遍棋盘上全部 64 个方格 2 马踏棋盘算法
  • Linux内核内存检测工具KASAN

    KASAN k z n KASAN 是 Kernel Address Sanitizer 的缩写 它是一个动态检测内存错误的工具 主要功能是检查内存越界访问和使用已释放的内存等问题 KASAN 集成在 Linux 内核中 随 Linux 内
  • linux内存调节之CMA

    本文贴代码过头了 以后想起来再优化一下吧 目录 概述 数据结构 构建初始化 DTS CONFIG DMA CMA 页表与物理页初始化 分配器激活 分配器使用 CMA部署 实战 概述 CMA Contiguous Memory Allocat
  • 数据结构之图的两种遍历实现(C语言版)

    上一期文章分享完了图的两种遍历方式 也是两种很重要的算法 DFS和BFS 这两种算法的应用和重要性我就不多说了 内行的人懂的都懂 今天这文章重要就是来上机实现这两种算法 又由于这两种算法都可以由邻接矩阵和邻接表来表示 博主分享的代码都是上机
  • 堆栈01--用两个栈实现队列

    堆栈01 用两个栈实现队列 jz05 题目概述 解析 参考答案 注意事项 说明 题目概述 算法说明 用两个栈来实现一个队列 完成队列的Push和Pop操作 队列中的元素为int类型 测试用例 队列先进先出 输入 1 2 输出 1 2 解析
  • 用指针访问一维数组

    文章目录 顺序查找 数组方式实现 指针实现方式 对一位数组元素的访问有三种方式 指针变量的关系运算 引例 数组实现方式 主函数 指针实现方式 主函数 一维数组作为函数的参数 实际应用 顺序查找 要求用指针实现 在整数集合r中顺序查找与给定值
  • RT-Thread 内存管理

    在计算机系统中 通常存储空间可以分为两种 内部存储空间和外部存储空间 内部存储空间通常访问速度比较快 能够按照变量地址随机访问 也就是我们通常所说的RAM 随机存储器 可以把它理解为电脑的内存 而外部存储空间内所保存的内容相对来说比较固定
  • 牛客剑指offer刷题其他算法篇

    文章目录 构建乘积数组 题目 思路 代码实现 第一个只出现一次的字符
  • 数组实现循环队列(增设队列大小size)

    目录 一 前言 1 如何实现循环 2 如何判断队列为空 3 如何判断队列为满 二 循环队列的结构定义 三 循环队列的创建及其初始化 四 入队 五 出队 六 取队头元素 七 取队尾元素 八 循环队列判空 九 循环队列判满 十 循环队列销毁 一
  • 【数据结构】单链表的定义和操作

    目录 1 单链表的定义 2 单链表的创建和初始化 3 单链表的插入节点操作 4 单链表的删除节点操作 5 单链表的查找节点操作 6 单链表的更新节点操作 7 完整代码 嗨 我是 Filotimo 很高兴与大家相识 希望我的博客能对你有所帮助
  • C++ AVL树(四种旋转,插入)

    C AVL树 四种旋转 插入 一 AVL树的概念及性质 二 我们要实现的大致框架 1 AVL树的节点定义 2 AVL树的大致框架 三 插入 1 插入逻辑跟BST相同的那一部分 2 修改平衡因子
  • 从源码角度来谈谈 HashMap

    HashMap的知识点可以说在面试中经常被问到 是Java中比较常见的一种数据结构 所以这一篇就通过源码来深入理解下HashMap 1 HashMap的底层是如何实现的 基于JDK8 1 1 HashMap的类结构和成员 HashMap继承
  • 最大流-Dinic算法,原理详解,四大优化,详细代码

    文章目录 零 前言 一 概念回顾 可略过 1 1流网络 1 2流 1 3最大流 1 4残留网络 1 5增广路

随机推荐