`
gaofen100
  • 浏览: 1186380 次
文章分类
社区版块
存档分类
最新评论

从 Qt 的 delete 说开来

 
阅读更多

在C++中学习过程中,我们都知道:

  • delete 和 new 必须 配对使用(一 一对应):delete少了,则内存泄露,多了麻烦更大。

Qt作为C++的库,显然是不会违背C++的前述原则的。可是:

  • 在Qt中,我们很多时候都疯狂地用new,却很少用delete,缺少的 delete 去哪儿了?!

注:本文暂不涉及智能指针(smart pointer)相关的东西,你可以考虑 Qt 智能指针学习 一文

Qt半自动的内存管理

在Qt中,以下情况下你new出的对象你可以不用 亲自去delete (但你应该清楚delete在何处被Qt调用的,怎么被调用的):

  • QObject及其派生类的对象,如果其parent非0,那么其parent析构时会析构该对象(本文内容围绕这一点展开 )

除此之外,有些类的对象可以接收设置一些特别的标记,比如:

  • QWidget及其派生类的对象,可以设置 Qt::WA_DeleteOnClose 标志位(当close时会析构该对象)
  • QAbstractAnimation派生类的对象,可以设置 QAbstractAnimation::DeleteWhenStopped
  • QRunnable::setAutoDelete()
  • MediaSource::setAutoDelete()
  • ...

注意:这些用法会有些陷阱 ,请注意看本文最后的3个小例子。

在Qt中,最基础和核心的类是:QObject 。它的魔力很大,本文只关注两点:

  • 父子关系
  • deleteLater

父子关系

在Qt中,每个 QObject 内部都有一个list,用来保存所有的 children,还有一个指针,保存自己的parent。当它自己析构时,它会将自己从parent的列表中删除,并且析构掉所有的children。

  • 注意:在 Qt 中,我们经常会遇到
    • 基类、派生类,或父类、子类。 这是对于派生体系来说的,和在C++相关书中看到的完全一样,与这的parent无关
    • 父对象、子对象、父子关系。 这是Qt中所特有的,也就是这儿的parent所引入的,与类的继承关系无关

建立与解除

Q_INVOKABLE QObject::QObject ( QObject * parent = 0 )
  • 创建一个QObject对象时,如果指定了父对象,它就会将自己添加到父对象的 children 列表中
QObject::~QObject () [virtual]
  • 当一个QObject对象析构时,它会将自己从父对象的 children 列表中移除(parent非0的话)
void QObject::setParent ( QObject * parent )
  • 通过该函数,将自己从原父对象的children中删除,添加到新parent的children列表中

注:这三个函数都是通过一个内部私有函数来实现的,这就是

QObjectPrivate::setParent_helper(QObject *o)

获取父、子对象

每个QObject只有一个父对象:

QObject * QObject::parent () const

子对象可以有多个

const QObjectList & QObject::children () const

所以可以根据条件来查找喽:

T QObject::findChild ( const QString & name = QString() ) const
QList<T> QObject::findChildren ( const QString & name = QString() ) const

deleteLater

deleteLater 包含两层意思了

  • delete
  • later

呵呵,似乎这是废话哈。

删除自己

在去年春节前的时候吧,有人对

obj-> deleteLater()

会像下面一样调用delete:

delete obj;

感到不解。然后我写了这样一个C++例子:

class A
{
  public:
  A(){}
  void deleteMe()
  {
      delete this;
  }
};

int main()
{
  A * a = new A;
  a->deleteMe();
  return 0;
} 

应该不需要解释吧

later

Qt 是事件驱动的,所以发送一个删除事件到事件系统就可以啦:

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

事件循环稍后看到该事件就会将其派发会这个widget:

bool QObject::event(QEvent *e)
{
    switch (e->type()) {
...
    case QEvent::DeferredDelete:
         ...

一些例子

无关痛痒?

很简短、很熟悉的一个例子是不?但是 如果你发现对象的析构函数始终不被成功调用 ,会有什么感觉?

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

这是 C++ GUI Programming with Qt 4 一书的第一个例子。我们注意到这儿的 label 既没有指定parent,也没有对其调用delete。

所以,这儿会造成内存泄露。

书中解释说,对于这种小例子,这点内存泄露不算什么。不清楚官方这个例子的意图是什么,或许是一开始就让大家用指针吧。

三种改进方式

  • 分配对象到stack而不是heap中
QLabel label("Hello Qt!");
label.show();
  • 设置标志位,这样,当我们点击关闭按钮时,close()函数将会调用deleteLater
label->setAttribute(Qt::WA_DeleteOnClose);
  • 动手调用delete(不就是少了一个么,我们补上还不行么)
int ret = app.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对象却不是通过new分配到heap中的。

为了使得用户减少自己显式使用delete,Qt将delete隐藏的比较深。这样一来,不使用new为对象分配空间时,反倒需要多多小心了。

隐蔽很深?

看个小例子:这个程序退出时会直接崩溃

#include <QtGui>
int main(int argc, char* argv[])
{
   QApplication app(argc, argv);
   QLabel label(tr"Hello Qt!");
   QWidget w;
   label.setParent(&w);
   w.show();
   return app.exec();
}
  • 问题出在哪儿呢?因为退出时,w 比 label 先被析构,当 w 被析构时,会删除chilren列表中的对象,也就是这儿的 label。但 label 却不是通过new分配在heap中,而是在stack中,可想而知,delete 一个再stack中的对象会怎么样了。相当于
QLabel label();
delete &label;
  • 两种改进办法:
    • 一是,将label分配到heap中
   QLabel *label = new QLabel("Hello Qt!");
   label.setParent(&w)
  • 再一种就是,确保label先于其parent被析构(调整一下顺序),这样,label析构时将自己从父对象的列表中移除自己,w析构时,children列表中就不会有分配在stack中的对象了。
   QWidget w;
   QLabel label(tr"Hello Qt!");

Qt 对象的父子关系的引入,简化了我们对内存的管理,但是,由于它会在你不太注意的地方调用 delete,所以,使用时还是要当心。

参考

分享到:
评论

相关推荐

    QT父子与QT对象delete

    当delete一个父对象的时候,Qt的对象模型机制保证了会自动的把 它的所有子对象,以及孙对象,等等,全部delete,从而保证不会有内存泄漏的情况发生。 任何事情都有正反两面作用,这种机制看上去挺好,但是却会对很...

    Qt http同步请求测试(restfull中常用的get/post/put/delete、文件上传/下载) 示例源码

    Qt http同步请求测试(restfull中常用的get/post/put/delete、文件上传/下载) 示例源码 https://blog.csdn.net/aggs1990/article/details/124104548 CSDN审核可能较慢,如无法下载,可以过段时间再回来看下

    Qt 入门中文资料 Qt从入门到精通

    c++ Qt中文入门教程,从Qt入门到Qt的深入研究。

    QT中文参考手册(QT help)

    从Qt 2.x移植到Qt 3.x 简体中文汉化日志 Qt季刊 API参考 模块 概述 所有的类 主要的类 分组的类 注释的类 继承关系 类表(图像) 所有的函数(很长) 头文件索引 PDF格式文档 man帮助格式文档 关于...

    qt-solutions-qt-solutions

    从Qt官网上下载的Qt的解决方案资料库(Qt Solutions Archive) http://qt.nokia.com/products/qt-addons/solutions-archive/ 有JPEG2000的,有Windows中Services的,有浏览器插件的,有单应用程序的,有SOAP的,有...

    QT 计算器QT 计算器QT 计算器QT 计算器

    REQT 计算器QT 计算器QT 计算器QT 计算器QT 计算器QT 计算器

    QT经典教程_详细一步一步例子

    为了使更多的Qt初学者能尽快入门Qt,也为了Qt及Qt Creator的快速普及,我们花费大量精力写出了这一系列教程。虽然教程的知识可能很浅显,虽然教程的语言可能不规范,但是它却被数十万网友所认可。我们会将这一系列...

    qt删除文件夹内所有内容

    qt删除文件夹内所有内容,包括文件和文件夹,逻辑还是有点小复杂的,已测试,木有问题

    qt自定义控件(开源)

    #ColorProgressBar 多彩进度条 编译环境: Qt5.x + mingw-QtCreater 运行截图: progressbarcolor为原版代码 #DynamicProgressBar 多态进度条 编译环境: Qt5.x + mingw-QtCreater 运行截图: #RoundProgressBar 编译...

    QT学习教程PDF(《Qt Creator快速入门》;C++ GUI Qt4 编程(第二版) ;Qt及Qt+Quick开发实战精解)

    QT学习教程PDF(《Qt Creator快速入门》;C++ GUI Qt4 编程(第二版) ;Qt及Qt+Quick开发实战精解) 三合一的压缩包,教程的源代码我整理出来后会上传

    屏蔽系统热键ctrl+alt+delete

    钩子类,屏蔽系统热键,应用系统api屏蔽系统功能键,ctrl+Tab, ctrl+alt, ctrl+alt+delete

    QT高级编程 中文版

    因此,即使是很有经验的qt程序开发人员,也可以从书中找出自己不曾注意到的技术点。书中的全部示例程序都已用qt 4.6或者qt 4.5在windows、mac os x和linux系统上进行了测试。 《qt高级编程》主要面向c++/qt程序开发...

    QT项目开发实例

    Qt是一个1991年由Qt Company开发的跨平台C++图形用户界面应用程序开发框架。它既可以开发GUI程序,也可用于开发非GUI程序,比如控制台工具和服务器。Qt是面向对象的框架,使用特殊的代码生成扩展(称为元对象编译器...

    Qt及QtQuick开发实战精解高清PDF

    Qt及Qt Quick开发实战精解,Qt及Qt Quick开发实战精解

    [Qt]Qt Creator汉化方法 qt汉化包

    D:\Qt\2010.04\share\qtcreator\translations目录下重启Qt Creator仍然可以汉化,qtcreator_zh_CN.qm的基础上进行汉化

    银行管理系统QT(c++)

    银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理系统QT(c++)银行管理...

    qt项目开发实例超详细版

    qt项目开发实例 Qt 是一个1991年由Qt Company开发的跨平台C++图形用户界面应用程序开发框架。它既可以开发GUI程序,也可用于开发非GUI程序,比如控制台工具和服务器。Qt是面向对象的框架,使用特殊的代码生成扩展...

    Qt 5.9 C++开发指南

    Qt是一套应用程序开发类库,但与MFC不同,Qt是跨平台的开发类库。Qt支持PC和服务器的平台,包括Windows、Linux、macOS等,还支持移动和嵌入式操作系统,如iOS、Embedded Linux、Android、WinRT等。跨平台意味着只需...

    解决qt-creator创建工程说“没有有效的qt版本问题”

    图解解决qt-creator遇到的创建工程时候出现“没有有效的qt版本问题”

Global site tag (gtag.js) - Google Analytics