Qt 之 new对象何时销毁

疑惑

在类的构造函数中,使用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);        
  • 4
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值