建议:在现代Qt开发中,应优先使用C++标准库的智能指针(std::shared_ptr、std::unique_ptr),仅在特定的Qt场景下使用Qt的智能指针(QSharedPointer、QScopedPointer)。
一 各智能指针对比
| 智能指针类型 | 所有权模型 | 使用场景 | 优势 | 劣势 |
|---|---|---|---|---|
std::unique_ptr |
独占所有权 | 明确的单一所有者 | 零开销、移动语义、与标准库完美集成 | 不能在Qt容器中直接使用 |
std::shared_ptr |
共享所有权 | 多个对象共享资源 | 线程安全引用计数、与标准库集成 | 引用计数开销 |
QScopedPointer |
独占所有权 | Qt风格的独占指针 | Qt父子关系自动管理 | 非标准、功能弱于std::unique_ptr |
QSharedPointer |
共享所有权 | Qt容器中的共享指针 | 与Qt容器无缝集成 | 非标准、仅在Qt环境中友好 |
二 Qt智能指针的历史背景
Qt早期(Qt4时代)提供了自己的智能指针类:
-
QScopedPointer:类似std::unique_ptr,用于管理动态分配的对象 -
QSharedPointer:类似std::shared_ptr,实现引用计数共享 -
QWeakPointer:类似std::weak_ptr,配合QSharedPointer使用
原因 :在C++11之前,标准库没有智能指针,Qt必须自己实现。但从C++11开始,标准库提供了完整的智能指针支持,且经过多年演进(C++14完善了std::unique_ptr,C++17/20进一步优化),现在标准库的智能指针已经成为事实标准。
三 具体使用建议
-
优先使用std::unique_ptr
-
在Qt容器中使用QSharedPointer
-
利用Qt的父子机制替代智能指针
-
特殊情况:跨线程传递对象
四 不建议使用QScopedPointer
-
功能弱于
std::unique_ptr:std::unique_ptr支持移动语义、自定义删除器、数组支持等 -
非标准:代码的可移植性差,与标准库代码集成困难
-
维护成本:未来Qt可能会逐步淘汰自己的智能指针
五 总结建议
| 场景 | 推荐方案 | 原因 |
|---|---|---|
| 管理非QObject对象 | std::unique_ptr |
零开销、标准、功能强大 |
| 多个所有者共享资源 | std::shared_ptr |
标准、线程安全 |
| QObject派生类(有父子关系) | 直接使用new + 父对象 |
Qt对象树自动管理 |
| Qt容器存储指针 | QSharedPointer |
与Qt容器值语义兼容 |
| 跨线程传递QObject | QSharedPointer |
线程安全的引用计数 |
| 自定义删除器(如数组) | std::unique_ptr<T[]> |
标准库完美支持 |
核心原则:
-
现代C++项目应优先使用标准库,除非遇到必须使用Qt特殊机制的场景
-
Qt的父子机制(QObject树) 比智能指针更适合管理QObject派生类
-
保持一致性:在团队项目中统一使用标准库智能指针,便于代码维护