shared_ptr是c++11中的智能指针,其可以自动的释放指针,避免了new之后忘记delete的问题。
shared_ptr 对象在内部指向两个内存位置:
1、指向对象的指针。
2、用于控制引用计数数据的指针。
计数数据指针的作用:
1、当新的 shared_ptr 对象与指针关联时,使用use_count(),其指针关联的引用计数增加1。
2、当任何 shared_ptr 对象超出作用域时,则在其析构函数中,它将关联指针的引用计数减1。如果引用计数变为0,则表示没有其他 shared_ptr 对象与此内存关联,在这种情况下,它使用delete函数删除该内存。
创建 shared_ptr 对象
- 使用原始指针创建 shared_ptr 对象
- 带有参数的 shared_ptr 构造函数是 explicit 类型的,所以不能像这样
std::shared_ptr<int> p1 = new int(),正确的创建方式如下:
std::shared_ptr<int> ptr2(new int());
此方法在堆上创建了两块内存:1:存储int
。2:控制块上用于引用计数的内存,管理附加此内存的 shared_ptr 对象的计数,最初计数将为1。
使用make_ptr创建空的 shared_ptr 对象
std::shared_ptr<int> ptr = std::make_shared<int>();
std::make_shared 一次性为int
对象和用于引用计数的数据都分配了内存,而new
操作符只是为int
分配了内存。
分离关联的指针
1.要使 shared_ptr 对象取消与相关指针的关联,可以使用reset()
函数:
();
2.可以直接设为nullptr
ptr = nullptr;
给shared_ptr添加自定义删除器
当 shared_ptr 对象调用其析构函数,它将引用计数减1,如果引用计数的新值为0,则删除关联的原始指针。析构函数中删除内部原始指针,默认调用的是delete()
函数。 当shared_ptr指向数组时,该函数针对数组类型创建的指针是不起作用的,需要自己来自定义一个删除器来delete[].
//删除器
void deleter(Sample * x)
{
std::cout << "delete\n";
delete[] x;
}
//第二个参数传递自定义删除器指针
std::shared_ptr<Sample> p(new Sample[2], deleter);
new和make_ptr的区别
- new是分配两次内存,make_ptr分配一次
- 在C++ 11中因为make_shared有std::move语义,在加上O2优化选项的时候,make_shared会比new快上将近1倍