【Qt】 也谈Qt的垃圾回收

阅读: 评论:0

【Qt】 也谈Qt的垃圾回收

【Qt】 也谈Qt的垃圾回收

前几天在做代码审核的时候,Kai Uwe Broulik建议使用QScopedPointer来替代手工内存管理,使用后发觉确实节约了不少代码量,我的CHERRY可以延长寿命了!但是通过简单地阅读代码,发现和Python等动态语言的垃圾回收机制不同,并没有使用引用计数!下面我们来看看:

Qt也有垃圾回收?

是的!从4.6开始,就可以替代:

if (myObj) {delete myObj;myObj = NULL;}

使用方法也很简单,在头文件里声明一个QScopedPointer<MyObject> myObjPtr;然后在源文件中初始化它:

MyClass::MyClass() : myObjPtr(new MyObject){// }

 

Qt怎么实现的?

查看QScopedPointer源代码里的注释:把基于栈的内存所有权给堆分配,称之为RAII。被这些拽术语的吓到了?那可以看看三点水的入门级《手把手教你构建 C 语言编译器》讲解的“虚拟”内存布局。QScopedPointer没有拷贝构造函数,也就是不能QScopedPointer(const QScopedPointer<T> &other),也没有重载赋值运算符,也就是不能myObjPtr = QScopedPointer<MyObject>(new MyObject);QScopedPointer模板类中定义了Cleanup的默认参数为QScopedPointerDeleter<T>,然后在QScopedPointer的析构函数里调用之Cleanup::cleanup,我们可以自定义一个故意不释放内存指针的MyScopedPointerDeleter模板类:


template <typename T>
struct MyScopedPointerDeleter
{static inline void cleanup(T *pointer){qDebug() << "33[31m" << "DEBUG:" << __FILE__ << __PRETTY_FUNCTION__ << "33[0m";}
};

那么再使用QScopedPointer<MyObject, MyScopedPointerDeleter<MyObject> >myObjPtr;时,相当于忘记了delete!针对new MyObject[XXX],QScopedArrayPointer模板类也提供了QScopedPointerArrayDeleter模板类的cleanup,针对C风格的malloc,还提供了QScopedPointerPodDeleter模板类的cleanup。

 

如何实现一个MyScopedPointer?

最简单的实现就是Ctrl+C/Ctrl+V:把qscopedpointer.h源代码拷贝出来,把QScoped替换成MyScoped,哦,对了,您不用再搬砖,我已经替大家搬完了!运行效果如所示:

 ./testscopedpointerDEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myobject.cpp MyObject::MyObject(QObject*)                                                                                             ...DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myobject.cpp MyObject::MyObject(QObject*)                                                                                             DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myobject.cpp virtual MyObject::~MyObject() ...DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myobject.cpp virtual MyObject::~MyObject() DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/testscopedpointer.cpp static void MyCustomScopedPointerDeleter::cleanup(T*) [with T = MyObject] DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myobject.cpp virtual MyObject::~MyObject() DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myscopedpointer.h static void MyScopedPointerDeleter::cleanup(T*) [with T = MyObject] DEBUG: /data/project/LeetcodeCN/leetcodecn.github.io/src/scopedpointer/myobject.cpp virtual MyObject::~MyObject() 

可以发现调用了3次MyObject的构造函数,但只成功(被MyScopedPointer的析构函数)调用了1次MyObject的析构函数,其他2次中一个是忘记delete了new分配的MyObject*,另一个是故意写一个忘记delete的MyCustomScopedPointerDeleter ;-)

 

其他的QXXXPointer

上面讲过QScopedPointer没有拷贝构造函数,没有重载赋值运算符,所以,无法使用QList<QScopedPointer<MyObject> >编译器会报错:使用了删除的(拷贝构造函数、赋值运算符)函数,这时就得使用QSharedPointer,它很像std::shared_ptr,都具有拷贝构造函数、重载赋值运算符。QSharedPointer可以通过toWeakRef转换成QWeakPointer,std::shared_ptr也可以转换成std::weak_ptr来检查对象是否被销毁。

本文发布于:2024-01-28 05:37:20,感谢您对本站的认可!

本文链接:https://www.4u4v.net/it/17063914445176.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

标签:也谈   垃圾   Qt
留言与评论(共有 0 条评论)
   
验证码:

Copyright ©2019-2022 Comsenz Inc.Powered by ©

网站地图1 网站地图2 网站地图3 网站地图4 网站地图5 网站地图6 网站地图7 网站地图8 网站地图9 网站地图10 网站地图11 网站地图12 网站地图13 网站地图14 网站地图15 网站地图16 网站地图17 网站地图18 网站地图19 网站地图20 网站地图21 网站地图22/a> 网站地图23