当前位置: 首页 > 编程日记 > 正文

Qt浅谈之一:内存泄露(总结)

一、简介
       Qt内存管理机制:Qt 在内部能够维护对象的层次结构。对于可视元素,这种层次结构就是子组件与父组件的关系;对于非可视元素,则是一个对象与另一个对象的从属关系。在 Qt 中,在 Qt 中,删除父对象会将其子对象一起删除。

C++中delete 和 new 必须配对使用(一 一对应):delete少了,则内存泄露,多了麻烦更大。Qt中使用了new却很少delete,因为QObject的类及其继承的类,设置了parent(也可在构造时使用setParent函数或parent的addChild)故parent被delete时,这个parent的相关所有child都会自动delete,不用用户手动处理。但parent是不区分它的child是new出来的还是在栈上分配的。这体现delete的强大,可以释放掉任何的对象,而delete栈上对象就会导致内存出错,这需要了解Qt的半自动的内存管理。另一个问题:child不知道它自己是否被delete掉了,故可能会出现野指针。那就要了解Qt的智能指针QPointer。

二、关联图

(1)Linux内存图,主要了解堆栈上分配内存的不同方式。

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

三、详解

1、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无关)。

2、内存问题例子

  • 例子一

  • #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();
    }

    分析:(1)label 既没有指定parent,也没有对其调用delete,所以会造成内存泄漏。书中的这种小例子也会出现指针内存的问题。
    改进方式:(1)分配对象到栈上而不是堆上

  • #include <QApplication>
    #include <QLabel>int main(int argc, char *argv[])
    {QApplication a(argc, argv);QLabel label("Hello Qt!");label.show();return a.exec();
    }

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

  • label->setAttribute(Qt::WA_DeleteOnClose);

    (3)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;
    }

    例子二

  • #include <QApplication>
    #include <QLabel>
    int main(int argc, char *argv[])
    {QApplication app(argc, argv);QLabel label("Hello Qt!");label.show();label.setAttribute(Qt::WA_DeleteOnClose);return app.exec();
    }

    运行:

    分析:程序崩溃,因为label被close时,delete &label;但label对象是在栈上分配的内存空间,delete栈上的地址会出错。
    有些朋友理解为label被delete两次而错误,可以测试QLabel label("Hello Qt!"); label.show();delete &label;第一次delete就会出错。

    例子三

    #include <QApplication>
    #include <QLabel>
    int main(int argc, char* argv[])
    {QApplication app(argc, argv);QLabel label("Hello Qt!");QWidget w;label.setParent(&w);w.show();return app.exec();
    }

  • 分析:Object内部有一个list,会保存children,还有一个指针保存parent,当自己析构时,会自己从parent列表中删除并且析构所有的children。
    w比label先被析构,当w被析构时,会删除chilren列表中的对象label,但label是分配到栈上的,因delete栈上的对象而出错。

    改进方式:(1)调整一下顺序,确保label先于其parent被析构,label析构时将自己从父对象的列表中移除自己,w析构时,children列表中就不会有分配在stack中的对象了。

  • #include <QApplication>
    #include <QLabel>
    int main(int argc, char* argv[])
    {QApplication app(argc, argv);QWidget w;QLabel label("Hello Qt!");label.setParent(&w);w.show();return app.exec();
    }

    (2)将label分配到堆上

    QLabel *label = new QLabel("Hello Qt!");
    label->setParent(&w)

    或者QLabel *label = new QLabel("Hello Qt!",this);

      • 例子四:野指针

  • #include <QApplication>
    #include <QLabel>
    int main(int argc, char* argv[])
    {QApplication app(argc, argv);QWidget *w = new QWidget;QLabel *label = new QLabel("Hello Qt!");label->setParent(w);w->show();delete w;label->setText("go");     //野指针return app.exec();
    }

    (上述程序不显示Label,仅作测试)

    分析:程序异常结束,delete w时会delete label,label成为野指针,调用label->setText("go");出错。

    改进方式:QPointer智能指针

  • #include <QApplication>
    #include <QLabel>
    #include <QPointer>
    int main(int argc, char* argv[])
    {QApplication app(argc, argv);QWidget *w = new QWidget;QLabel *label = new QLabel("Hello Qt!");label->setParent(w);QPointer<QLabel> p = label;w->show();delete w;if (!p.isNull()) {label->setText("go");}return app.exec();
    }

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

    发送一个删除事件到事件系统:

  • void QObject::deleteLater()
    {QCoreApplication::postEvent(this, new QEvent(QEvent::DeferredDelete));
    }

    3、智能指针
            如果没有智能指针,程序员必须保证new对象能在正确的时机delete,四处编写异常捕获代码以释放资源,而智能指针则可以在退出作用域时(不管是正常流程离开或是因异常离开)总调用delete来析构在堆上动态分配的对象。

    Qt家族的智能指针:

(1)QPointer

QPointer是一个模板类。它很类似一个普通的指针,不同之处在于,QPointer 可以监视动态分配空间的对象,并且在对象被 delete 的时候及时更新。

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

    // QPointer 表现类似普通指针 QDate *mydate = new QDate(QDate::currentDate()); QPointer mypointer = mydata; mydate->year();    // -> 2005 mypointer->year(); // -> 2005 // 当对象 delete 之后,QPointer 会有不同的表现 delete mydate; if(mydate == NULL) printf("clean pointer"); else printf("dangling pointer"); // 输出 dangling pointer if(mypointer.isNull()) printf("clean pointer"); else printf("dangling pointer"); // 输出 clean pointer

(2)std::auto_ptr

    // QPointer 表现类似普通指针 QDate *mydate = new QDate(QDate::currentDate()); QPointer mypointer = mydata; mydate->year();    // -> 2005 mypointer->year(); // -> 2005 // 当对象 delete 之后,QPointer 会有不同的表现 delete mydate; if(mydate == NULL) printf("clean pointer"); else printf("dangling pointer"); // 输出 dangling pointer if(mypointer.isNull()) printf("clean pointer"); else printf("dangling pointer"); // 输出 clean pointe

auto_ptr被销毁时会自动删除它指向的对象。
std::auto_ptr<QLabel> label(new QLabel("Hello Dbzhang800!"));

(3)其他的类参考相应文档。

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

#include <QApplication>
#include <QObjectCleanupHandler>
#include <QPushButton>int main(int argc, char* argv[])
{QApplication app(argc, argv);// 创建实例QObjectCleanupHandler *cleaner = new QObjectCleanupHandler;// 创建窗口QPushButton *w = new QPushButton("Remove Me");w->show();// 注册第一个按钮cleaner->add(w);// 如果第一个按钮点击之后,删除自身
   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);QObject::connect(w, SIGNAL(clicked()), cleaner, SLOT(deleteLater()));w->show();return app.exec();
}

在上面的代码中,创建了三个仅有一个按钮的窗口。第一个按钮点击后,会删除掉自己(通过 deleteLater() 槽),此时,cleaner 会自动将其从自己的列表中清除。第三个按钮点击后会删除 cleaner,这样做会同时删除掉所有未关闭的窗口。

(2)引用计数
  应用计数是最简单的垃圾回收实现:每创建一个对象,计数器加 1,每删除一个则减 1。

class CountedObject : public QObject 
{ Q_OBJECT 
public: CountedObject() { ctr=0; } void attach(QObject *obj) { ctr++; connect(obj, SIGNAL(destroyed(QObject*)), this, SLOT(detach())); } public slots: void detach() { ctr--; if(ctr <= 0) delete this; } private: int ctr; 
};

利用Qt的信号槽机制,在对象销毁的时候自动减少计数器的值。但是,我们的实现并不能防止对象创建的时候调用了两次attach()。

(3)记录所有者

更合适的实现是,不仅仅记住有几个对象持有引用,而且要记住是哪些对象。例如:

 class CountedObject : public QObject { public: CountedObject() {} void attach(QObject *obj) { // 检查所有者 if(obj == 0) return; // 检查是否已经添加过 if(owners.contains(obj)) return; // 注册 
         owners.append(obj); connect(obj, SIGNAL(destroyed(QObject*)), this, SLOT(detach(QObject*))); }  public slots: void detach(QObject *obj) { // 删除 
         owners.removeAll(obj); // 如果最后一个对象也被 delete,删除自身 if(owners.size() == 0) delete this; }   private: QList owners; 
};

现在我们的实现已经可以做到防止一个对象多次调用 attach() 和 detach() 了。然而,还有一个问题是,我们不能保证对象一定会调用 attach() 函数进行注册。毕竟,这不是 C++ 内置机制。有一个解决方案是,重定义 new 运算符(这一实现同样很复杂,不过可以避免出现有对象不调用 attach() 注册的情况)。
四、总结
      Qt 简化了我们对内存的管理,但是,由于它会在不太注意的地方调用 delete,所以,使用时还是要当心。

五、参考

http://doc.qt.nokia.com/4.7/qobject.html

http://www.cuteqt.com/blog/?p=824

http://blog.csdn.net/dbzhang800/article/details/6300025
http://blog.csdn.net/dbzhang800/article/details/6403285
http://devbean.blog.51cto.com/448512/526734/

相关文章:

LINUX新手入门-1.装系统

LINUX新手入门-1.装系统首先我们用虚拟机模拟 装linux系统&#xff0c;然后下一步下一步&#xff0c;然后完成后&#xff0c;编辑一些设置&#xff0c;把镜像放上面就可以了选第一项&#xff0c;安装系统&#xff0c;查看镜像是否能运行&#xff0c;直接跳过&#xff0c;选择语…

Log4cplus1.04的使用

首先&#xff0c;从http://sourceforge.net/projects/log4cplus/files/log4cplus-stable/下载最新的版本&#xff0c;解压缩&#xff0c;用vs2008打开msvc8文件夹下的log4cplus.sln&#xff0c;并按照提示转换。在Solution Configurations下拉列表框中&#xff0c;会有Debug、D…

FRVT赛程全纪录:格灵深瞳全球排名前五

作者 | 张德兵&#xff0c;格灵深瞳首席科学家&算法部负责人来源 | 转载自知乎张德兵最近两个月&#xff0c;格灵深瞳首席科学家&算法部负责人张德兵与算法团队参加了全球人脸识别算法测试(FRVT、Face Recognition Vendor Test)。虽然是第一次参加此比赛&#xff0c;格…

反转比特位(文章最后有干货)【转】

转自&#xff1a;https://blog.csdn.net/wuxianglonghaohao/article/details/21602305 http://www.newhottopic.com/2014/03/20/reverse-bits/ 把一个无符号整数的比特位反转顺序。有很多种方法来实现这个。我们这里给出一个算法&#xff1a;通过异或运算来交换&#xff0c;然后…

过关斩将打进Kaggle竞赛Top 0.3%,我是这样做的

作者 | Lavanya Shukla译者 | Monanfei责编 | 夕颜出品 | AI科技大本营&#xff08;id&#xff1a;rgznai100&#xff09;导读&#xff1a;刚开始接触数据竞赛时&#xff0c;我们可能会被一些高大上的技术吓到。各界大佬云集&#xff0c;各种技术令人眼花缭乱&#xff0c;新手们…

JavaBean规范

2019独角兽企业重金招聘Python工程师标准>>> &#xff08;1&#xff09;JavaBean 类必须是一个公共类&#xff0c;并将其访问属性设置为 public &#xff08;2&#xff09;JavaBean 类必须有一个空的构造函数&#xff1a;类中必须有一个不带参数的公用构造器&#x…

vigra1.8.0的使用

VIGRA stands for "Vision with Generic Algorithms". Its a novel computer vision library that puts its main emphasis oncustomizablealgorithms and data structures. 1、首先&#xff0c;从http://hci.iwr.uni-heidelberg.de/vigra/下载最新源代码&#xff0…

17个Python小窍门

python中相对不常见却很实用的小窍门。 空谈不如来码代码吧&#xff1a; 交换变量值 给列表元素创建新的分隔符 找列表中出现次数最多的元素 核对两个字符是否为回文 反向输出字符串 反向输出列表 转置2维数组 链式比较 我刚整理了一套2018最新的0基础入门和进阶教程&#xff0…

用产品思路建设中台,这走得通吗?| 白话中台

作者 | 王健&#xff0c;ThoughtWorks首席咨询师。 十多年国内外大型企业软件设计开发&#xff0c;团队组织转型经验。一直保持着对技术的热爱&#xff0c;热衷于技术分享。目前专注在企业平台化转型、中台战略规划&#xff0c;微服务架构与实施&#xff0c;大型遗留系统服务化…

利用cvMinAreaRect2求取轮廓最小外接矩形

转自&#xff1a;http://blog.csdn.net/mine1024/article/details/6044856 对给定的 2D 点集&#xff0c;寻找最小面积的包围矩形&#xff0c;使用函数&#xff1a; CvBox2D cvMinAreaRect2( const CvArr* points, CvMemStorage* storageNULL ); points 点序列或点集数组 …

电脑开机显示Invalidsystemdisk

开机或重启无法进入系统&#xff0c;并在屏幕上显示Invalidsystemdisk&#xff0c;Replacethediskandthenpressanykey或者diskerror之类的字样&#xff0c;这是怎么回事&#xff0c;该如何解决&#xff1f;今天u大师就为大家解决下。 出现这个原因是因为现在的电脑没有可以启…

Windows7 64位下vs2008配置OpenCV2.3.1

1、下载OpenCV2.3.1&#xff1a;http://www.opencv.org.cn/index.php/Download&#xff1b; 2、下载后解压缩&#xff1a;OpenCV-2.3.1-win-superpack.exe&#xff0c;生成一个opencv文件夹&#xff1b; 3、下载CMake&#xff1a;http://www.cmake.org/cmake/resources/softw…

腾讯拥抱开源:首次公布开源路线图,技术研发向共享、复用和开源迈进

整理 | 夕颜出品 | AI科技大本营&#xff08;ID:rgznai100&#xff09;导读&#xff1a;去年&#xff0c;知乎上一篇讨论腾讯技术的帖子异常火爆&#xff0c;讨论的主题是当下&#xff08;2018 年&#xff09;腾讯的技术建设是否处于落后同体量公司的状态&#xff0c;这篇帖子得…

Babylon.js 3.3发布:更强大的粒子系统和WebVR支持

Babylon.js 3.3版本利用微软混合现实工具包&#xff08;MRTK&#xff09;的功能来改进WebVR开发&#xff0c;并改进了其粒子系统控件。 MRTK提供了一系列脚本和组件来加速混合现实应用程序的开发。为了简化GUI VR构建&#xff0c;Bablyon.js利用3D体积网格来布局VR场景的界面&a…

基于Erlang语言的视频相似推荐系统 | 深度

作者丨gongyouliu来源 | 转载自大数据与人工智能&#xff08;ID:ai-big-data&#xff09;【导语】&#xff1a;作者在上一篇文章《基于内容的推荐算法》中介绍了基于内容的推荐算法的实现原理。在本篇文章中作者会介绍一个具体的基于内容的推荐算法的实现案例。该案例是作者在2…

MinGW简介

转自&#xff1a;http://baike.baidu.com/view/98554.htm MinGW是指只用自由软件来生成纯粹的Win32可执行文件的编译环境&#xff0c;它是Minimalist GNU on Windows的略称。这里的“纯粹”是指使用msvcrt.dll的应用程序。无法使用MFC (Microsoft Foundation Classes微软基础类…

Confluence 6 创建小组的公众空间

2019独角兽企业重金招聘Python工程师标准>>> 现在是我们可以开始创建公众空间的时候了&#xff0c;全世界都希望知道这个项目和勇敢的探险活动。 在这个步骤中&#xff0c;我们将会创建一个项目小组的空间&#xff0c;并且将这个空间公布给全世界。这个表示的是你将…

windows 7 可以清除的文件

缓解系统磁盘空间不足的情况1、系统盘根目录下的MSOCache是office的安装备份文件&#xff0c;可以删除。2、c:\user\用户名\appdate\local\temp是软件安装时留下的临时文件。3、c:\windows\SoftwareDistribution中存放的是系统补丁更新包及旧的系统文件。4、c:\windows\winsxs\…

阿里最新论文解读:考虑时空域影响的点击率预估模型DSTN

作者 | 石晓文转载自小小挖掘机&#xff08;ID: wAIsjwj&#xff09;【导语】&#xff1a;在本文中&#xff0c;阿里的算法人员同时考虑空间域信息和时间域信息&#xff0c;来进行广告的点击率预估。什么是时空域&#xff1f;我们可以分解为空间域(spatial domain)和时间域(tem…

windows7 64位机上配置MinGW+Codeblocks+ wxWidgets

在Windows7 64位机子上安装配置MinGWCodeblockswxWidgets步骤如下&#xff1a; 1、 下载mingw-get-inst-20111118&#xff1a;http://sourceforge.net/projects/mingw/&#xff1b; 2、 双击mingw-get-inst-20111118.exe&#xff0c;一般按默认即可&#xff0c;选择自己需要…

jQuery带动画的弹出对话框

在线演示 本地下载

陶哲轩实分析 习题 13.4.6

设 $(X,d)$ 是度量空间,并设 $(E_{\alpha})_{\alpha\in I}$ 是 $X$ 中的一族连通集合.还设 $\bigcap_{\alpha\in I}E_{\alpha}$ 不空.证明 $\bigcup_{\alpha\in I}E_{\alpha}$ 是连通的.证明:由于 $\bigcap_{\alpha\in I}E_{\alpha}$ 是不空的,因此存在 $p\in \bigcap_{\alpha\…

一年参加一次就够,全新升级的AI开发者大会议程出炉!

“只讲技术&#xff0c;拒绝空谈”的AI开发者大会再次来临&#xff01;2018 年的AI开发者大会&#xff0c;作为年度人工智能领域面向专业开发者的一次高规格技术盛会&#xff0c;上千名开发者与上百名技术专家齐聚一堂&#xff0c;大会以“AI技术与应用”为核心&#xff0c;就人…

Windows7下配置MinGW+CodeBlocks+OpenCV2.3.1

1、下载mingw-get-inst-20111118&#xff1a;http://sourceforge.net/projects/mingw/&#xff1b; 2、双击mingw-get-inst-20111118.exe&#xff0c;一般按默认即可&#xff0c;选择自己需要的组件&#xff1b; 3、添加MinGW环境变量&#xff1a;选择计算机-->点击右键--…

Spark SQL基本操作以及函数的使用

2019独角兽企业重金招聘Python工程师标准>>> 引语&#xff1a; 本篇博客主要介绍了Spark SQL中的filter过滤数据、去重、集合等基本操作&#xff0c;以及一些常用日期函数&#xff0c;随机函数&#xff0c;字符串操作等函数的使用&#xff0c;并列编写了示例代码&am…

OpenCV提取轮廓(去掉面积小的轮廓)

转自&#xff1a;http://www.kaixuela.net/?p23 #include <stdio.h> #include "cv.h" #include "cxcore.h" #include "highgui.h" #include <iostream> using namespace std; #pragma comment(lib,"cv.lib") #pra…

软工作业 5:词频统计——增强功能

一、基本信息 1.1 编译环境、项目名称、作者 1 #编译环境:python3.6 2 #项目名称&#xff1a;软工作业5-词频统计—增强功能 3 #作者&#xff1a;1613072055 潘博 4 # 1613072056 侯磊 1.2项目地址 本次作业地址: https://www.cnblogs.com/panboo/项目git地址: https://g…

Linux之文件权限管理

chmod ux转载于:https://www.cnblogs.com/chaoren399/archive/2013/03/11/2953727.html

如果三十年前有这些AI技术,可可西里的悲剧不会发生

作者 | 神经小姐姐来源 | HyperAI超神经&#xff08;ID&#xff1a;HyperAI&#xff09;而被盗猎者大量的非法捕杀。多种野生动物都处于濒临灭绝的局面&#xff0c;人工智能等技术&#xff0c;能够在帮助保护野生动物上&#xff0c;发挥比较大的作用&#xff0c;让我们能够生存…

Percona-Server-5.5.30安装

1、安装系统环境 yum install -y gcc gcc-c autoconf automake zlib* libxml* ncurses-devel libmcrypt* libtool-ltdl-devel* cmake bison 2、下载源码包 1 http://www.percona.com/downloads/ 2 3 wget -c http://www.percona.com/redir/downloads/Percona-Server-5.5/Perc…