当前位置: 首页 > news >正文

【面试八股总结】C++11新特性:智能指针

参考资料 :阿秀、代码随想录

        智能指针是一个类,用于存储指向动态分配对象的指针负责自动释放动态分配的对象防止堆内存泄露。动态分配的资源,交给一个类对象去管理,当类对象声明周期结束时,自动调用析构函数释放资源。

        C++11提供了两种智能指针,两种智能指针的区别在于管理底层指针的方式shared_ptr允许多个指针指向同一个对象;unique_ptr则“独占”所指向的对象。标准库还定义了一个名为weak_ptr的伴随类,是一种弱引用,指向shared_ptr所管理的对象。

一、 shared_ptr

        采⽤引用计数器的方法,允许多个智能指针指向同⼀个对象,每当多⼀个指针指向该对象时,指向该对象的所有智能指针内部的引用计数加1,每当减少⼀个智能指针指向对象时,引用计数会减1,当计数为0的时候会自动的释放动态分配的资源

1. 使用方法

  • 创建智能指针,需要给出指针可以指向的类型
// 创建智能指针
shared_ptr<string> p1;       // 指向string
shared_ptr<list<int>> p2;    // 指向int的list
  •  分配和使用动态内存 make_shared<T>(args)
shared_ptr<string> p3 = make_shared<string>(10,'9');    // 指向一个值为“9999999999”的字符串

2. 线程安全问题

(1)同⼀个shared_ptr被多个线程“读”是安全的;

(2)同⼀个shared_ptr被多个线程“写”是不安全的;

(3)共享引用计数的不同的shared_ptr被多个线程”写“ 是安全的。

二、 unique_ptr

        采用独享所有权语义,⼀个非空的unique_ptr总是拥有它所指向的资源。转移⼀个unique_ptr将会把所有权全部从源指针转移给目标指针,源指针被置空所以unique_ptr不支持普通拷贝和赋值操作,不能用在 STL标准容器中;局部变量的返回值除外(因为编译器知道要返回的对象将要被销毁);如果你拷贝⼀个unique_ptr,那么拷贝结束后,这两个unique_ptr都会指向相同的资源,造成在结束时对同⼀内存指针多次释放而导致程序崩溃。

1. 创建unique_ptr

        定义unique_ptr时,需要将其绑定到一个new返回的指针上,初始化unique_ptr必须采用直接初始化形式。

unique_ptr<string> p1;           
unique_ptr<int> p2(new int(42)); // p2指向一个值为42的int

2. realese/reset

        虽然不能够拷贝或赋值unique_ptr,但是可以通过调用release和reset将指针的所有权从一个(非const)unique_ptr转移给另一个unique;

unique_ptr<string> p2(p1.release());    // release将p1置空unique_ptr<string> p3(new string("Chesnut"));  
// 将所有权从p3转移给p2
p2.reset(p3.release());    // reset释放p2原本指向空间

3. 传递unique_ptr参数和返回unique_ptr

        不能拷贝unique_ptr的规则有一个例外: 可以拷贝或赋值一个将要被销毁的unique_ptr。最常见的例子就是从函数返回一个unique_ptr。

unique_ptr<int> clone(int p){return unique_ptr<int>(new int(p));
}

三、 weak_ptr 弱引用

        引用计数有⼀个问题就是互相引⽤形成环(环形引用),这样两个指针指向的内存都无法释 放,需要使用weak_ptr打破环形引用

        weak_ptr是⼀个弱引用,它是为了配合shared_ptr而引⼊的⼀种智能指针,它指向⼀个由shared_ptr管理的对象而不影响所指对象的生命周期,也就是说,它只引用,不计数。如果⼀块 内存被shared_ptr和weak_ptr同时引⽤,当所有shared_ptr析构了之后,不管还有没有weak_ptr引⽤该内存,内存也会被释放。所以weak_ptr不保证它指向的内存⼀定是有效的在使用之前使⽤函数lock()检查weak_ptr是否为空指针。

http://www.lryc.cn/news/348129.html

相关文章:

  • 【教程向】从零开始创建浏览器插件(二)深入理解 Chrome 扩展的 manifest.json 配置文件
  • 美易官方:美国房地产贷款逾期率飙升,银行业危机仍可控?现货黄金暂守2360
  • SwiftUI中的@StateObject和@ObservedObject的区别
  • 类与对象(二)
  • LeetCode/NowCoder-链表经典算法OJ练习2
  • 英伟达解码性能NVDEC
  • 文心一言 VS 讯飞星火 VS chatgpt (255)-- 算法导论18.3 1题
  • C++ | Leetcode C++题解之第73题矩阵置零
  • 用 Supabase CLI 进行本地开发环境搭建
  • 三极管 导通条件
  • 一次pytorch分布式训练精度调试过程
  • STM32(GPIO)
  • python设计模式---观察者模式
  • 【论文笔记】KAN: Kolmogorov-Arnold Networks 全新神经网络架构KAN,MLP的潜在替代者
  • 【投稿资讯】区块链会议CCF C -- CoopIS 2024 截止7.10 附录用率
  • React Native 之 开发环境搭建(一)
  • DS高阶:B树系列
  • 第五百零三回
  • [动态规划] 完美覆盖
  • redis深入理解之实战
  • python设计模式---工厂模式
  • 探索Vue 3.0中的v-html指令
  • anaconda 环境配置
  • DS:顺序表、单链表的相关OJ题训练(2)
  • 上传到 PyPI
  • 盛最多水的容器(双指针)
  • 【深度学习】实验3 特征处理
  • MoneyPrinter国内版改造
  • C++ 派生类的引入与特性
  • Poe是什么?怎样订阅Poe?