疑惑
在类的构造函数中,使用new创建了一个button, 当关闭这个窗口时, new出来的内存会释放吗?
MyWindow::MyWindow(QWidget *parent, const char *name,WFlags f)
: QMainWindow(parent,name,f | WDestructiveClose)
{
QPushButton *button = new QPushButton("hello",this);
}
对象树机制
QT的父子对象机制是在 QWidget和QOject中实现的。当我们使用父对象来创建一个对象的时候 ,父对象会把这个 对象添加到自己的子对象列表中。当这个父对象被删除的时候,它会遍历它的子对象类表并且删除每一个子对象,然后子对象们自己再删除它们自己的子对象,这样递归调用直到所有对象都被删除。
这种父子对象机制会在很大程度上简化我们的内存管理工作,减少内存泄露的风险。我们需要显式地删除(就是用delete)的对象是那些使用new创建的并且没有父对象的对象。如果我们在删除一个对象的父对象之前删除它,QT会自动地从它的父对象的子对象列表中移除它的。
该机制确保对象中已接管的其他对象被正确析构。
Qt半自动的内存管理
以下情况下,new出的对象,可以不用亲自去delete
- QObject及其派生类的对象,如果其parent非0,那么其parent析构时会析构该对象
- 该对象的父类不为NULL
父子关系
- 父对象
- 子对象
- 父子关系
建立与解除
- 创建一个QObject对象时,如果指定了父对象,它就会将自己添加到父对象的children列表中
Q_INVOKABLE QObject::QObject ( QObject * parent = 0 )
- 当一个QObject对象析构时,它会将自己从父对象的 children 列表中移除(parent非0的话)
QObject::~QObject () [virtual]
- 将自己从原父对象的children中删除,添加到新parent的children列表中
void QObject::setParent ( QObject * parent )
deleterLater
#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 Qt4 第一个例子,但是QLabel没有指定parent, 也没有对齐调用delete, 所以这里会造成内存泄漏。
三种改进方式
- 分配对象到stack而不是heap中
QLabel label("Hello Qt!");
label.show();
- 设置标志位,当点击关闭按钮时, close()函数会调用deleterlater
label->setAttribute(Qt::WA_DeleteOnClose);
- 手动调用调用delete
int ret = app.exec();
deleter 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();
}
另外一个例子
#include <QtGui>
int main(int argc, char* argv[])
{
QApplication app(argc, argv);
QLabel label(tr"Hello Qt!"); // label申明在父对象之前
QWidget w; // 父对象
label.setParent(&w); // label设置父对象为w
w.show();
return app.exec();
}
- 问题
上述程序退出时,会奔溃。 原因: w比label先析构,当w被析构时,会删除children列表中的对象,也就是这儿的label。 但是label在stack中,因此delete 一个位于stack中的对象,会出错。相当于
QLabel label();
delete &label;
- 两种改进
QLabel *label = new QLabel("Hello Qt!"); //将label分配到heap中
label.setParent(&w)
QWidget w;
QLabel label(tr"Hello Qt!"); //确保label先于其parent被析构,w析构时,children列表中就不会有分配在stack中的对象了
label.setParent(&w);