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
#include
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
#include
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QLabel label("Hello Qt!");
label.show();
return a.exec();
}
B、设置标志位,close()后会delete label。
#include
#include
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
#include
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
#include
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
#include
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
#include
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
#include
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 判断就为空了。
2、std::auto_ptr
auto_ptr被销毁时会自动删除它指向的对象。
五、垃圾回收机制
1、QObjectCleanupHandler
Qt 对象清理器是实现自动垃圾回收的很重要部分。QObjectCleanupHandler可以注册很多子对象,并在自己删除的时候自动删除所有子对象。同时,它也可以识别出是否有子对象被删除,从而将其从它的子对象列表中删除。QObjectCleanupHandler类可以用于不在同层次中的类的清理操作,例如,当按钮按下时需要关闭很多窗口,由于窗口的 parent 属性不可能设置为别的窗口的 button,此时使用QObjectCleanupHandler类就会很方便。
#include
#include
#include
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对象。