智能指针(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。)
- QObject及其派生类的对象,如果其parent非0,那么其parent析构时会析构该对象。
- QWidget及其派生类的对象,可以设置 Qt::WA_DeleteOnClose 标志位(当close时会析构该对象)。
- QAbstractAnimation派生类的对象,可以设置 QAbstractAnimation::DeleteWhenStopped。
- QRunnable::setAutoDelete()、MediaSource::setAutoDelete()。
- 父子关系:父对象、子对象、父子关系。这是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中,当然会出问题。运行之后正常,但关闭窗口会报错。
解决方法:
- 把QLabel和QWidget的构造语句交换一下;
- 让QLabel对象创建在heap上;