智能指针
其实一直都对智能指针的应用场景不清楚,项目中也很少用到,今天在 @zccrs 大佬的帮助下,大概理解了智能指针的作用和应用场景。
设计思想
智能指针依赖一种叫引用计数的手段来协助管理对象指针,通过引用计数为0时删除对象指针来完成内存的释放,本质上是通过栈对象来管理堆对象的一种方法。
传统做法
void test() { |
当出现异常时,delete将不会被执行到,t也就泄露了。虽然我们可以在异常那里把delete给加上,但是在较为大型的项目中,如果对代码进行review来排查这种错误,将会是非常麻烦的一件事,所以为了避免内存泄漏,发明了基于引用技术的智能指针。
智能指针做法
void test() { |
如果不关心std::unique_ptr是什么,这段代码无意是糟糕的,new出来的Test对象根本没有地方被删除,内存泄露了。
但是不必担心,指针已经由std::unique_ptr来管理了,根本不会发生内存泄漏,对象将在离开函数作用域以后被删除。
这就是智能指针的方便之处。
智能指针的基本实现
智能指针都通过模板编程来实现,模板是C++的另一大功能,可以使我们更关心实现而不需要关心具体的对象,通过更加抽象的方式来编写程序。
智能指针有两层,里层用来保存对象的指针和引用计数,外层用来调用里层来控制引用计数。
里层的辅助类
template<typename T> |
外层的控制类
template<typename T> |
通过重写控制类的拷贝构造函数和赋值运算符重载来更新引用计数。
使用实例
void test() { |
这样我们就有一个简单的智能指针了,不过他还存在一些问题,比如循环引用导致内存泄漏,没有->和*的操作运算符等。所以我们需要更强大的智能指针来帮助我们。
几种智能指针的介绍
标准库提供了几个针对不同方面使用的智能指针,以满足我们的需求。
- unique_ptr
只允许一个所有者,除非确信你需要共享该指针,则应该使用
shared_ptr
。可以转移到新的所有者,但是不会复制和共享。 - shared_ptr
采用引用计数的智能指针,如果你想将一个原始指针分配给多个所有者,请使用该智能指针,直到
shared_ptr
所有者超出了范围或放弃所有权,才会删除原始指针,大小为两个指针,一个用于对象,一个用于引用计数。 - weak_ptr
结合
shared_ptr
使用的特殊智能指针,提供一个或多个shared_ptr
实例所拥有的对象的访问,但是不会增加引用计数。如果你想观察某个对象,但是不需要保持活动状态,则可以使用该智能指针。在某些情况下,需要断开shared_ptr
实例间的循环引用。
如何正确的选择智能指针
智能指针只需要区分需不需要共享使用,如果外部需要使用这个对象,使用shared_ptr
,否则就使用unique_ptr进行独占使用。
陷阱和坑
- 不要使用相同的内置指针来初始化多个智能指针
- 不要主动回收智能指针内原始指针的内存
- 不要使用智能指针的get来初始化或者reset另一个智能指针
- 智能指针管理的资源只会默认删除new分配的内存,如果不是new分配的,则需要使用删除器