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对象

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Qt是一款跨平台的C++应用开发框架,它提供了一系列的工具和方法来帮助开发者排查内存泄漏问题。 一、使用内存泄漏检测工具:Qt提供了自己的内存泄漏检测工具,例如QtCreator中的内存分析器。这个工具可以帮助开发者跟踪Qt对象的内存使用情况,找出潜在的内存泄漏问题。 二、覆盖对象的析构函数:在Qt中,可以通过重载对象的析构函数来释放对象所占用的资源。在析构函数中,可以手动删除动态分配的内存释放其他资源,以确保对象的销毁不导致内存泄漏。 三、使用智能指针:Qt中的QSharedPointer和QWeakPointer是用于管理动态分配内存的智能指针类。使用这些智能指针可以自动管理内存释放,避免了手动释放内存的步骤,从而减少内存泄漏的风险。 四、检查对象的生命周期:仔细检查对象的创建和销毁过程,确保对象在不再使用时被正确释放。特别注意在使用Qt信号槽机制时,确保槽函数没有未释放的对象引用。 五、使用QScopedPointer管理局部对象:QScopedPointer是Qt提供的一个用于管理局部对象的类,它在作用域结束时自动调用析构函数,从而释放对象所占用的内存。使用QScopedPointer可以避免忘记释放局部对象的问题。 六、使用工具进行内存泄漏监控:除了Qt自带的内存分析器外,还可以使用一些第三方工具,例如Valgrind和Dr. Memory等,来帮助检测和分析应用程序中的内存泄漏问题。 在开发过程中,如果发现内存泄漏问题,可以结合上述工具和方法进行排查和解决。及时发现和修复内存泄漏问题,可以提高应用程序的性能和稳定性。 ### 回答2: Qt 内存泄漏是指在使用Qt框架开发程序时,由于代码中存在错误或者不当的使用方式导致内存无法被正确释放,从而造成内存泄漏问题。针对Qt内存泄漏的排查手段如下: 1. 使用内存泄漏检测工具:可以使用一些专门的内存泄漏检测工具,例如Valgrind、Dr. Memory等,对Qt程序进行检测。这些工具可以报告内存泄漏的具体位置和调用栈信息,有助于快速定位问题。 2. 注意资源的正确释放:在编写代码时,需要特别注意Qt对象的生命周期和正确释放资源的方法。使用了堆分配的Qt对象,如QObject的子类,需要手动释放内存。正确使用delete或deleteLater来删除对象,确保资源得到正确释放。 3. 使用自动析构对象:使用Qt的RAII机制,合理使用QScopedPointer、QSharedPointer等智能指针类,可以自动管理内存释放,减少手动释放的疏忽和错误。 4. 给关键对象设置父对象:对于Qt对象,没有父对象的对象将不在父对象被销毁时自动销毁。因此,应及时为需要销毁的对象设置正确的父对象,以保证它们能够被正确释放。 5. 检查定时器:如果程序中使用了定时器,需要仔细检查定时器的启动和停止机制,确保在对象被删除时能够正确地停止和释放相关的定时器。 6. 追踪对象创建和销毁:在程序运行过程中,通过打印相关的信息或者重写Qt对象的创建和析构函数,可以追踪对象的创建和销毁过程,从而找到潜在的内存泄漏问题。 综上所述,Qt内存泄漏的排查手段包括使用内存泄漏检测工具、注意资源的正确释放、使用自动析构对象、为关键对象设置父对象、检查定时器以及追踪对象的创建和销毁等方法。通过这些手段,可以有效地排查和解决Qt程序中的内存泄漏问题。 ### 回答3: Qt是一个跨平台的C++框架,开发者可以使用它来开发图形用户界面(GUI)应用程序和各种软件。在使用Qt开发过程中,内存泄漏是一个常见的问题。为了排查和解决内存泄漏问题,可以采用以下手段: 1. 使用工具:Qt提供了一些工具来帮助排查内存泄漏问题。其中最重要的是`valgrind`,它是一个内存调试和分析工具。使用valgrind可以检测程序运行过程中的内存泄漏和非法内存访问情况。 2. 使用Qt自带的内存管理工具:Qt提供了一些内存管理工具,比如`QScopedPointer`和`QSharedPointer`。这些工具可以帮助开发者自动管理内存资源,减少内存泄漏的风险。使用这些工具来替代常规的指针操作可以极大地减少内存泄漏的概率。 3. 代码审查:定期对代码进行审查,尤其是在使用动态内存分配的地方。确保在每次动态分配内存后都有对应的释放操作,不要忘记及时释放动态分配的内存。 4. 合理使用QObject继承体系:在Qt中,QObject继承体系具有自动删除子对象的功能。合理使用QObject继承体系可以使对象的删除更加安全和方便。避免手动删除已经添加到父对象的子对象,这样可以减少因为忘记删除而引起的内存泄漏。 5. 使用自动化测试:编写包含内存泄漏检测的自动化测试用例。通过这些测试用例,可以及早发现潜在的内存泄漏问题,并在开发过程中及时修复。 总之,排查和解决Qt内存泄漏问题需要综合使用工具、合理使用内存管理工具、代码审查、合理使用QObject继承体系以及使用自动化测试等手段。这些方法可以帮助开发者及时发现和解决内存泄漏问题,提高程序的健壮性和性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值