好长一段时间没明白共享指针的理解和使用,今天认认真真查了一些资料,搞懂了很多。在这里整理了一下两个链接的内容。
主要参考链接:
智能指针之共享指针shared_ptr 的理解、使用(全)_爱水人家的博客-CSDN博客_共享指针的使用
一、shared_ptr类
shared_ptr 主要的功能是,管理动态创建的对象的销毁。它的基本原理就是记录对象被引用的次数,当引用次数为 0 的时候,也就是最后一个指向某对象的共享指针析构的时候,共享指针的析构函数就把指向的内存区域释放掉。
与普通指针相比,共享指针对象重载了 * 、-> 和==运算符, 所以你可以像通常的指针一样使用它。没有重载+、-、++、--、[ ]等运算法。
1.默认初始化的智能指针中保存着一个空指针。
2.智能指针的使用方式与普通指针类似。解引用一个智能指针返回它指向的对象。
1.make_shared函数
最安全的分配和使用动态内存的方法是调用一个名为make_shared的标准库函数。当要用make_shared时,必须指定想要创建的对象的类型。定义方式与模板类相同,在函数名之后跟一个尖括号,在其中给出类型:
//指向一个值为42的int的shared_ptr
shared_ptr<int> p3 = make_shared<int>(42);
检查 shared_ptr 对象的引用计数
p1.use_count();
创建空的 shared_ptr 对象
因为带有参数的 shared_ptr 构造函数是 explicit 类型的,所以不能像这样std::shared_ptr<int> p1 = new int();隐式调用它构造函数。创建新的shared_ptr对象的最佳方法是使用std :: make_shared:
std::shared_ptr<int> p1 = std::make_shared<int>();
std::make_shared 一次性为int对象和用于引用计数的数据都分配了内存,而new操作符只是为int分配了内存。
2.shared_ptr 的“赋值”
shared_ptr 也可以直接赋值,但是必须是赋给相同类型的 shared_ptr 对象,而不能是普通的 C 指针或 new 运算符的返回值。当共享指针 a 被赋值成 b 的时候,如果 a 原来是 NULL, 那么直接让 a 等于 b 并且让它们指向的东西的引用计数加 1; 如果 a 原来也指向某些东西的时候,如果 a 被赋值成 b, 那么原来 a 指向的东西的引用计数被减 1, 而新指向的对象的引用计数加 1. 就是说以下代码是允许的:
std::shared_ptr<T> a(new T());
std::shared_ptr<T> b(new T());
a = b; // 此后 a 原先所指的对象会被销毁,b 所指的对象引用计数加 1
shared_ptr 的对象在构造之后,可以被赋予空值,此时使用的应该是 reset() 函数或者nullptr,如:
std::shared_ptr<T> a(new T());
a.reset(); // 此后 a 原先所指的对象的引用计数-1,并且 a 会变成 NULL。这里内存会被销毁
a=nullptr; //同上。推荐多用
当然理论上也可以这样写:
std::shared_ptr<T> a(new T());
a = std::shared_ptr<T>(); // ,相当于给 a 赋值一个新构造的 无名shared_ptr对象, 也就是 NULL
二、总结
1.shared_ptr一般用make_shared初始化
2.shared_ptr 可以当作函数的参数传递,或者当作函数的返回值返回,这个时候其实也相当于使用复制构造函数,指向的对象的引用计数加1。
3.离开作用域(对象销毁)或者给shared_ptr赋予一个新值时,指向的对象的引用计数减1。
4.调用reset或者指向nullptr,原来所指的对象会被销毁