C++/Qt中的内存管理

智能指针(Smart Pointer)和原始指针(Raw Pointer)

  • 智能指针负责对象生命期管理(这里假设智能指针作为类的非静态成员变量,并借助类的构造函数和析构函数来完成动态对象的自动化管理):所以动态对象的创建和析构全都由unique_ptr和shared_ptr来做;
  • 原始指针不负责对象生命周期管理:原始指针擅长调用动态对象,原因就是简化接口。如果这时候使用shared_ptr来传递动态对象和使用原始指针来传递动态对象本质上没有区别,为了简单还是传递原始指针更好一些;

Qt的半自动化的内存管理

在Qt中,QObject的类及其继承的类,如果设置了parent(也可在构造时使用setParent函数或parent的addChild),当parent被delete时,这个parent的相关所有child都会自动delete,不用用户手动处理。(QObject内部有一个list,会保存children,还有一个指针保存parent,当自己析构时,会自己从parent列表中删除并且析构所有的children。)

  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半自动内存机制的缺陷在于parent不区分它的child是分配在stack上还是heap上,结果会出现parent析构stack上的子对象。因此,Qt对象析构的核心原则是保证子对象先析构,这样它会把自己从父对象列表中删除,二者取消关联,那么父对象析构时就不会再次析构子对象了。

例:

QApplication app(argc, argv);
QLabel label("Hello Qt!");
QWidget w;
label.setParent(&w);
w.show();
return app.exec();

错误原因:

    C++中,本地对象的析构函数的调用顺序与他们的构造顺序相反。所以是QWidget对象先析构,也就是父对象先析构,它会删除子对象label,但label却不是通过new分配在heap中,而是在stack中,当然会出问题。运行之后正常,但关闭窗口会报错。

解决方法:

  1. 把QLabel和QWidget的构造语句交换一下;
  2. 让QLabel对象创建在heap上;
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值