【C++】常用的智能指针及 std::unique_ptr 和 std::shared_ptr 应用

C++ 智能指针是C++标准库提供的类模板,用于自动管理动态分配的对象的生命周期。它们旨在解决传统原始指针可能导致的内存泄漏和资源未正确释放等问题,通过封装对动态内存的访问和控制,实现自动化的资源清理。以下是对C++中几种主要智能指针类型的详细介绍:

1. std::unique_ptr (C++11)

原理与特点:

表现为独占所有权(exclusive ownership)的智能指针。任何时候只有一个unique_ptr实例可以拥有并管理某个对象。

  • 当unique_ptr离开其作用域或被显式重置时,它会自动删除其所指向的对象。
  • 不支持复制构造函数和赋值操作符,但可以通过转移语义(move semantics)进行移动构造和移动赋值,转移过程中原unique_ptr变为空指针,所有权转移到新unique_ptr。
  • 可以直接或间接管理非数组类型以及数组类型。

成员函数:

  • release():释放所有权,返回指向对象的原始指针,之后unique_ptr变为空。
  • reset():释放当前所拥有的对象(如果存在),可选地接受一个新的裸指针来接管。
  • get():返回指向对象的原始指针,但不改变所有权。
  • operator->() 和 operator*():提供对托管对象的直接访问。

案例

#include <memory>

// 1.创建一个指向整型对象的unique_ptr,默认使用delete运算符释放资源
std::unique_ptr<int> uptr(new int(10));

// 2.C++ 14 使用 std::make_unique
std::unique_ptr<int> uptr = std::make_unique<int>(10);
std::unique_ptr<MyClass> uptr = std::make_unique<MyClass>();
// 3.转移所有权
int* raw_ptr = new int(10);
std::unique_ptr<int> uptr(raw_ptr); 
// 如果uptr已有对象,先释放旧对象,再转移所有权
uptr.reset(raw_ptr);

2. std::shared_ptr (C++11)

原理与特点:

实现共享所有权(shared ownership)的智能指针。多个shared_ptr实例可以同时指向并共享同一个对象。

  • 内部维护一个引用计数(reference count)。每当新的shared_ptr拷贝或赋值现有shared_ptr时,引用计数递增;当shared_ptr销毁或重置时,引用计数递减。当引用计数降至零时,自动删除所管理的对象。
  • 支持弱引用计数,通过std::weak_ptr关联到同一对象,但不会增加引用计数。

成员函数:

  • use_count():返回当前共享对象的引用计数值。
  • reset():释放当前所拥有的对象(如果存在),可选地接受一个新的裸指针来接管。
  • get():返回指向对象的原始指针,但不改变所有权。
  • operator->() 和 operator*():提供对托管对象的直接访问。
    swap():交换两个shared_ptr的内容。
    与std::make_shared一起使用可以提高效率和减少内存碎片。

3. std::weak_ptr (C++11)

原理与特点:

  • 提供对std::shared_ptr管理对象的弱引用,不增加引用计数。
  • 可以检查所指向的对象是否仍然存活(未被删除)。
  • 通过调用lock()方法临时获取一个可访问对象的shared_ptr,如果对象已被删除,则返回空shared_ptr。

成员函数:

  • expired():返回对象是否已被删除(即引用计数是否为零)。
  • lock():尝试获取一个指向对象的shared_ptr,如果对象仍然有效,则返回非空shared_ptr,否则返回空shared_ptr。

基本用法

//1 创建
int main() {
    // 创建一个 shared_ptr 并指向新建的 MyClass 对象
    std::shared_ptr<MyClass> ptr1 = std::make_shared<MyClass>(42);

    // 访问对象的方法
    std::cout << "Value: " << ptr1->GetValue() << std::endl;

    // 创建另一个 shared_ptr,共享 ptr1 所指向的对象
    std::shared_ptr<MyClass> ptr2(ptr1);

    // 当 ptr1 和 ptr2 都离开作用域时,MyClass 对象会被自动释放
}
// 2 共享
int main() {
    std::shared_ptr<MyClass> ptr = std::make_shared<MyClass>(13);

    // 将 ptr 传递给函数,共享所有权
    PrintValue(ptr);

    // 即使 PrintValue 结束,对象也不会被释放,因为 ptr 仍在作用域内
}

4. std::auto_ptr (已弃用, C++98)

注意:

std::auto_ptr是C++98引入的智能指针类型,但在C++11中已被废弃,取而代之的是std::unique_ptr。由于其转移所有权的行为不符合现代C++的预期(如不支持容器),不再推荐使用。

5. 综上

C++智能指针根据不同的所有权语义分为unique_ptr(独占所有权)、shared_ptr(共享所有权)和weak_ptr(弱引用),它们分别适用于不同的资源管理场景,有助于编写更安全、更易于维护的C++代码。使用智能指针时应遵循其设计意图,正确处理所有权转移和生命周期管理,避免循环引用等潜在问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/573254.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【每日刷题】Day22

【每日刷题】Day22 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. 1669. 合并两个链表 - 力扣&#xff08;LeetCode&#xff09; 2. 11. 盛最多水的容器 - 力扣&#…

分类算法——ROC曲线与AUC指标(九)

知道TPR与FPR TPRTP/(TP FN) 所有真实类别为1的样本中&#xff0c;预测类别为1的比例 FPR FP/(FP TN) 所有真实类别为0的样本中&#xff0c;预测类别为1的比例 ROC曲线 ROC曲线的横轴就是FPRate&#xff0c;纵轴就是TPRate&#xff0c;当二者相等时&#xff0c;表示的意义…

Linux 内核设备树 ranges属性

今天有人问了我一下ranges属性&#xff0c;找了相关资料确认后&#xff0c;记录一下&#xff1a; 参考资料链接&#xff1a;让你完全理解linux内核设备树ranges属性地址转换 - vkang - 博客园 (cnblogs.com) ranges属性定义如下&#xff1a; ranges < local_address pa…

webpack面试题(持续汇总ing。。。)

webpack的编译过程 初始化 此阶段&#xff0c;webpack会将CLI参数、配置文件、默认配置进行融合&#xff0c;形成一个最终的配置对象。对配置的处理过程是依托一个第三方库 yargs 完成的。此阶段相对比较简单&#xff0c;主要是为接下来的编译阶段做必要的准备目前&#xff0c;…

三数之和 ---- 双指针

题目链接 题目: 分析: 解法一: 暴力解法, 将所有的三元组都算出来看是否为0, 题目要求去重操作, 所以我们可以使用set去重解法二: 因为我们知道当计算两数之和时, 我们使用的方法是将数组排序,然后利用"双指针"那么同理, 计算三个数之和: 1. 排序2. 固定一个数a, …

数据库管理-第176期 浅析代码团队建设(20240425)

数据库管理176期 2024-04-25 数据库管理-第176期 浅析代码团队建设&#xff08;20240425&#xff09;1 国内现状2 需求管控3 竞争与迭代总结 数据库管理-第176期 浅析代码团队建设&#xff08;20240425&#xff09; 作者&#xff1a;胖头鱼的鱼缸&#xff08;尹海文&#xff09…

安卓Activity的setContentView()流程分析

目录 前言一、Activity的视图加载过程1.1 视图结构1.2 流程分析1.2.1 Activity.java -->setContentView()1.2.2 Activity.java -->getWindow()1.2.3 PhoneWindow.java -->setContentView()1.2.4 PhoneWindow.java --->installDecor()1.2.4.1 PhoneWindow.java ---&…

Yolov5 export.py实现onnx模型的导出

查了很多资料&#xff0c;很多用python代码写的&#xff0c;只需要这个库那个库的&#xff0c;最后都没成功。 不如直接使用Yolov5里面的 export.py实现模型的转换。 一&#xff1a;安装依赖 因为yolov5里面的requirments.txt是将这些转换模型的都注释掉了 所以需要解除注释…

SpringCloud alibaba整合OpenFeign

目录 一、为什么使用OpenFeign 二、准备两个服务 三、最简单使用- 返回字符串 ①引入openfeign依赖 ②调用端在启动类上添加EnableFeignClients注解 ③在被调用端写一个简单的接口 ④在调用端新建一个service类 专门用于远程调用 ​编辑 ⑤ 在调用端写一个conteoller …

翻译《The Old New Thing》 - What does SHGFI_USEFILEATTRIBUTES mean?

What does SHGFI_USEFILEATTRIBUTES mean? - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20040601-00/?p39073 Raymond Chen 2004年06月01日 在使用 SHGetFileInfo 函数时&#xff0c;你可以设置一个名为 SHGFI_USEFILEATTRIBUTES 的标志…

绿联搭建rustdesk服务器

绿联搭建rustdesk服务器&#xff0c;不再使用向日葵 注意&#xff1a;本服务器需要有动态公网IP以及自己的域名&#xff0c;ipv6未测试。 1. 拉取镜像 rustdesk/rustdesk-server-s6:latest 注意是这个-s6的镜像。 2. 部署镜像 2.1 内存配置 本服务器比较省内存&#xff0…

区块链安全应用-------压力测试

基于已有的链进行测试&#xff08;build_chain默认建的链 四个节 点&#xff09;&#xff1a; 第一步&#xff1a;搭链 1. 安装依赖 在ubuntu操作系统中&#xff0c;操作步骤如下&#xff1a; sudo apt install -y openssl curl 2. 创建操作目录, 下载安装脚本 ## 创建操作…

力扣数据库题库学习(4.22日)

577. 员工奖金 问题链接 思路分析 Employee表与Bonus表通过empId字段可以连接&#xff0c;需求是查出奖金少于1000的员工名和奖金值。 这里奖金少于1000的情况就是没有奖金有奖金但少于1000 这里我给出的解决方案就是使用左连接&#xff0c;将Employee表作为左表&#xff…

js的算法-交换排序(冒泡)

交换排序 所谓交换排序&#xff0c;是指根据序列中两个元素关键字的比较结果来对换这两个记录在序列中的位置。基于交换的排序算法很多&#xff0c;本次介绍冒泡排序和快速排序。 冒泡 基本思想 从后往前&#xff08;或从前往后&#xff09;两两比较相邻元素的值&#xff0…

Nginx第3篇-使用ngx_http_proxy_connect_module配置https正向代理

场景 我使用python爬虫&#xff0c;然后需要个代理&#xff0c;所以就用Nginx搭了一个代理服务器。对Nginx也不太熟&#xff0c;慢慢摸索&#xff0c;搭建完之后发现只能代理http的请求&#xff0c;无法穿透https。几经折腾和摸索发现一个强大的HTTP代理模块&#xff1a;ngx_h…

泛微OA对接北森HR系统场景解析

随着企业信息化建设的深入推进&#xff0c;跨系统集成已成为提升管理效率、实现数据一体化的关键举措。详细阐述其如何通过泛微OA&#xff08;Office Automation&#xff09;系统与北森HR&#xff08;Human Resource&#xff09;系统的深度对接&#xff0c;实现人员信息、员工请…

RIP最短路实验(思科)

华为设备参考&#xff1a;RIP最短路实验&#xff08;华为&#xff09; 一&#xff0c;技术简介 RIP&#xff08;Routing Information Protocol&#xff0c;路由信息协议&#xff09;是一种基于距离矢量的内部网关协议&#xff0c;工作原理是每个路由器周期性地向邻居路由器发…

深度解析 Spring 源码:揭秘BeanFactory 之谜

文章目录 一、认识BeanFactory1.1 BeanFactory的概述1.2 BeanFactory与 ApplicationContext的区别 二、BeanFactory源码解读2.1 BeanFactory 接口2.1.1 getBean()2.1.2 containsBean()2.1.3 isSingleton() 2.2 DefaultListableBeanFactory 类2.2.1 registerBeanDefinition()2.2…

游戏行业干货科普 | 各个诚实都有哪些游戏公司?

本文主要列举上海、北京、广州、深圳、成都、杭州等城市游戏公司名称&#xff0c;大家可以码住&#xff0c;慢慢看~ 上海 米哈游 游戏势力新一极&#xff0c;近十年唯一一家打破腾讯、网易二强局面的公司莉莉丝 卡牌自研头部&#xff0c;SLG发行头部&#xff0c;最懂商业化的…

创建Maven项目的时候让选择maven模板

创建Maven项目的时候让选择maven模板 心得 工欲利其事 必先利其器。如果你想要干成一件事 那么必须先要精通对应的工具使用。之前我不太注重工具 我觉得只要代码写的好就可以了 但是当我们了解了产品经理的一些思想之后&#xff0c;我才明白一个好的产品是可以给用户提供多大…