【C++进阶篇】智能指针
C++内存管理终极指南:智能指针从入门到源码剖析
- 一. 智能指针
- 1.1 auto_ptr
- 1.2 unique_ptr
- 1.3 shared_ptr
- 1.4 make_shared
- 二. 原理
- 三. shared_ptr循环引用问题
- 三. 线程安全问题
- 四. 内存泄漏
- 4.1 什么是内存泄漏
- 4.2 危害
- 4.3 避免内存泄漏
- 五. 最后
一. 智能指针
智能指针通过RAII(Resource Acquisition Is Initialization)机制,将内存管理封装为类生命周期行为,其核心价值体现在:
- 自动内存回收:通过析构函数自动释放资源,避免忘记delete导致的内存泄漏
- 异常安全性:在异常抛出时仍能保证资源释放
- 所有权语义:明确资源归属关系,减少悬垂指针风险
1.1 auto_ptr
特点:拷贝时将被拷贝对象的资源转移给拷贝对象,会导致被拷贝对象悬空问题,访问会崩溃。**建议:**坚决不要使用该智能指针。
1.2 unique_ptr
特点:见名知意,不支持拷贝,只支持移动。
使用场景:
- 当某种特定场景不需要拷贝,强烈建议使用它。
int main()
{unique_ptr<int> sp(new int[10]);//unique_ptr<int> sp1 = sp;int* fp = sp.get();//对该指针进行操作for (size_t i = 0; i < 10; i++){fp[i] = i + 1;}for (size_t i = 0; i < 10; i++){cout << fp[i] << " ";}cout << endl;cout << "sp交换前: ";cout << "sp _ptr:" << sp.get() << endl;unique_ptr<int> sp1;sp1.swap(sp);cout << "sp交换后: ";cout << "sp _ptr:" << sp.get() << endl;cout << "sp1 _ptr:" << sp1.get() << endl;sp1.release();//将指针置空cout << "调用release()后: ";cout << "sp1 _ptr:" << sp1.get() << endl;return 0;
}
- 输出结果:
从结果可以看出调用swap后资源的管理权转移给调用者对象,自己置空。
1.3 shared_ptr
特点:支持拷贝也支持移动。底层是使用计数方式来看看是哪个对象来释放和清理资源。
- 使用场景:
当某种场景需要拷贝时,推荐使用它。
struct Date
{int _year;int _month;int _day;Date(int year = 1, int month = 1, int day = 1):_year(year), _month(month), _day(day){cout << "_year = " << _year << " _month = " << _month << " _day = " << _day << endl;}~Date(){cout << "~Date()" << endl;}
};int main()
{shared_ptr<Date> sp(new Date(2025, 6, 9));cout << sp.use_count()<<endl;shared_ptr<Date> sp1 = sp;//赋值cout << sp.use_count() << endl;//引用计数cout << sp.get() << endl;//获取原生指针cout << sp1.get() << endl;cout << sp.operator->() << endl;return 0;
}
- 输出结果:
从结果可以看出创建出的新对象与拷贝对象指向的资源一致。
sp->_year;
上面语句访问是允许的。因为智能指针里面重载operator ->() 同时返回的是管理对象的类的对象。
上述语句等价于:shared_ptr通过重载operator->,使得sp->_day的语法等价于**(sp)._day*。
下面再看其它问题:
shared_ptr<Date> sp1(new Date[10]);
程序会崩溃!!!
输出结果:
- 原因分析:
- shared_ptr默认使用delete释放内存(针对单个对象)但new Date[ ],必须使用delete [ ] 释放资源。错误类型匹配会导致未定义行为。
这里的T是Date实例化时,而构造时Date[ ],导致析构时资源不匹配。
解决办法:
使用自定义(推荐):说白了就是让智能指针管理资源的对象类型与构造时的对象数据类型一致。就可以解决了。
shared_ptr<Date[]> sp1(new Date[10]);
输出结果:
从结果可以看出构造时的10个Date对象都被正确清理了。
其它的方法可以使用仿函数对象,lambda表达式,函数指针对象等构造时将删除器传递给智能指针整个类,删除器在类内部初始化了,因为这个类内部包含指向对象指针,然后释放和清理资源。
// lambda表达式做删除器
auto delArrOBJ = [](Date* ptr) {delete[] ptr; };
unique_ptr<Date, decltype(delArrOBJ)> up4(new Date[5], delArrOBJ);
shared_ptr<Date> sp4(new Date[5], delArrOBJ);
// 函数指针做删除器
template<class T>
void DeleteArrayFunc(T* ptr)
{delete[] ptr;
}
unique_ptr<Date, void(*)(Date*)> up3(new Date[5], DeleteArrayFunc<Date>);
shared_ptr<Date> sp3(new Date[5], DeleteArrayFunc<Date>);
1.4 make_shared
std::make_shared 是 C++11 引入的工厂函数,用于高效、安全地创建 shared_ptr 智能指针。
基本用法:
#include <memory>// 创建 shared_ptr<int>
auto sp1 = std::make_shared<int>(42);// 创建 shared_ptr<Date>
class Date { /* ... */ };
auto sp2 = std::make_shared<Date>(2025, 6, 9);
性能优化:一次内存分配
传统方式:
shared_ptr sp(new Date(2025,6,9)); // 两次内存分配
- 第一次分配:为 Date 对象分配内存
- 第二次分配:为引用计数控制块分配内存
make_shared 方式:
auto sp = std::make_shared(2025,6,9); // 一次内存分配
- 底层原理:内存分配策略
- 调用 ::operator new(sizeof(T) + sizeof(ControlBlock))
- 将对象和控制块放置在同一块连续内存中
二. 原理
auto_ptr 转移资源,思路不被认可,而unique_ptr 不支持拷贝,思路较简单,下面重点看看shared_ptr 设计思路 及 原理。
- 思路:
主要这⾥⼀份资源就需要⼀个引⽤计数,所以引⽤计数才⽤静态成员的⽅式是⽆法实现的,要使⽤堆上动态开辟的⽅式,构造智能指针对象时来⼀份资源,就要new⼀个引⽤计数出来。多个shared_ptr指向资源时就++引⽤计数,shared_ptr对象析构时就–引⽤计数,引⽤计数减到0时代表当前析构的shared_ptr是最后⼀个管理资源的对象,则析构资源。
- 问题:为啥shared_ptr计数需要再堆上开空间,静态方式行不行???
不行,因为需要特定的实例对象指向资源时,计数器才+1,因为需要一个资源共享一个计数器。
假如使用静态方式:
static int static_counter = 0; // 错误!所有对象共享同一个计数器
导致虽然是类对象实例,但未指向该资源,导致计数器逻辑错误。
模拟实现shrared_ptr智能指针:
namespace W
{template<class T>class shared_ptr{public:explicit shared_ptr(T* ptr = nullptr):_ptr(ptr), _pcount(new int(1)){}template<class D>explicit shared_ptr(T* ptr = nullptr,D del):_ptr(ptr), _pcount(new int(1)),_del(del){}shared_ptr(const shared_ptr<T>& sp):_ptr(sp._ptr), _pcount(sp._pcount),_del(sp._del){++(*_pcount);}void release(){if (--(*_pcount) == 0){//delete _ptr;_del(_ptr);delete _pcount;_ptr = nullptr;_pcount = nullptr;}}shared_ptr<T> operator=(const shared_ptr<T>& sp){if (this != &sp){release();_ptr = sp._ptr;_pcout = sp._pcount;++(*_pcount);}return *this;}~shared_ptr(){release();}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}T* get()const{return _ptr;}int use_count()const{return *_pcount;}operator bool(){return _ptr != nullptr;}private:T* _ptr;int* _pcount;std::function<void(T*)> _del = [](T* ptr) {delete ptr; };//默认删除器};
}
三. shared_ptr循环引用问题
循环引用导致资源未被释放,从而导致内存泄漏,使用weak_ptr可以解决该问题,减少引用个数。
- 下面以一个场景来看看循环引用导致内存泄漏场景:
当n1和n2析构时,计数器分别减1,节点中的指针分别指向对方,导致每个资源引用计数器增加1。
n1节点中的next指针指向n2,n2节点中的prev指针指向n1,next什么时候析构呢,等着n2的prev指针不再指向是就析构了,n2的prev指针什么时候析构呢,等着n1的next不在指向时,就析构了。又回到原始问题,这就导致内存泄漏。
使用 weak_ptr 可以解决问题。
std::weak_ptr<ListNode> _next;
std::weak_ptr<ListNode> _prev;
下面详细介绍一下weak_ptr原理
- 功能:
weak_ptr是C++11引入的智能指针,主要用于解决shared_ptr的循环引用问题,并提供对共享资源的非拥有式观察。
- 非拥有式观察
weak_ptr不增加对象的引用计数,仅观察由shared_ptr管理的资源。它通过共享控制块(Control Block)跟踪对象状态,但不会影响对象生命周期。
- 解决循环引用
当两个对象通过shared_ptr互相引用时,引用计数无法归零,导致内存泄漏。将其中一个引用改为weak_ptr可打破循环。例如:
class B;
class A {
public:std::shared_ptr<B> b_ptr;
};
class B {
public:std::weak_ptr<A> a_ptr; // 使用weak_ptr打破循环
};
- expired():快速检查对象是否存活(无需创建shared_ptr)。
总结:
weak_ptr通过非拥有式观察机制,有效解决了shared_ptr的循环引用问题,并支持缓存、观察者模式等场景。理解其控制块共享、引用计数管理及安全访问方法,能帮助开发者编写更健壮的C++代码。
三. 线程安全问题
如果多个线程在堆上同时进行对该计数器进行操作,就会导致线程安全问题。
解决办法:
- 加互斥锁。
- 将计数器不设置为int* 类型,而设置为atomic*。
atmoic<int>* _pcount;
四. 内存泄漏
4.1 什么是内存泄漏
内存泄漏指因为疏忽或错误造成程序未能释放已经不再使⽤的内存,⼀般是忘记释
放或者发⽣异常释放程序未能执⾏导致的。内存泄漏并不是指内存在物理上的消失,⽽是应⽤程序分
配某段内存后,因为设计错误,失去了对该段内存的控制,因⽽造成了内存的浪费。
4.2 危害
普通程序运⾏⼀会就结束了出现内存泄漏问题也不⼤,进程正常结束,⻚表的映射
关系解除,物理内存也可以释放。⻓期运⾏的程序出现内存泄漏,影响很⼤,如操作系统、后台服
务、⻓时间运⾏的客⼾端等等,不断出现内存泄漏会导致可⽤内存不断变少,各种功能响应越来越
慢,最终卡死。
4.3 避免内存泄漏
⼯程前期良好的设计规范,养成良好的编码规范,申请的内存空间记着匹配的去释放。ps:这个理
想状态。但是如果碰上异常时,就算注意释放了,还是可能会出问题。需要下⼀条智能指针来管理
才有保证。
- 尽量使⽤智能指针来管理资源,如果⾃⼰场景⽐较特殊,采⽤RAII思想⾃⼰造个轮⼦管理。
- 定期使⽤内存泄漏⼯具检测,尤其是每次项⽬快上线前,不过有些⼯具不够靠谱,或者是收费。
- 总结⼀下:内存泄漏⾮常常⻅,解决⽅案分为两种:1、事前预防型。如智能指针等。2、事后查错型。如泄漏检测⼯具。
五. 最后
本文深入探讨了C++智能指针(auto_ptr、unique_ptr、shared_ptr、weak_ptr)的原理与应用。智能指针通过RAII机制实现自动内存管理,提升代码健壮性。unique_ptr独占资源,shared_ptr共享资源并通过引用计数管理生命周期,weak_ptr则提供非拥有式观察以解决循环引用问题。文章还介绍了make_shared的高效内存分配策略,并强调了线程安全与内存泄漏防范的重要性,是C++开发者掌握现代内存管理的实用指南。关于C++全内容到此结束了,下面将进入Linux网络篇的学习征程。
相关文章:

【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...
站群服务器的应用场景都有哪些?
站群服务器主要是为了多个网站的托管和管理所设计的,可以通过集中管理和高效资源的分配,来支持多个独立的网站同时运行,让每一个网站都可以分配到独立的IP地址,避免出现IP关联的风险,用户还可以通过控制面板进行管理功…...
scikit-learn机器学习
# 同时添加如下代码, 这样每次环境(kernel)启动的时候只要运行下方代码即可: # Also add the following code, # so that every time the environment (kernel) starts, # just run the following code: import sys sys.path.append(/home/aistudio/external-libraries)机…...
C#学习第29天:表达式树(Expression Trees)
目录 什么是表达式树? 核心概念 1.表达式树的构建 2. 表达式树与Lambda表达式 3.解析和访问表达式树 4.动态条件查询 表达式树的优势 1.动态构建查询 2.LINQ 提供程序支持: 3.性能优化 4.元数据处理 5.代码转换和重写 适用场景 代码复杂性…...
【JavaSE】多线程基础学习笔记
多线程基础 -线程相关概念 程序(Program) 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序,比如我们使用QQ,就启动了一个进程,操作系统就会为该进程分配内存…...

接口自动化测试:HttpRunner基础
相关文档 HttpRunner V3.x中文文档 HttpRunner 用户指南 使用HttpRunner 3.x实现接口自动化测试 HttpRunner介绍 HttpRunner 是一个开源的 API 测试工具,支持 HTTP(S)/HTTP2/WebSocket/RPC 等网络协议,涵盖接口测试、性能测试、数字体验监测等测试类型…...

STM32HAL库USART源代码解析及应用
STM32HAL库USART源代码解析 前言STM32CubeIDE配置串口USART和UART的选择使用模式参数设置GPIO配置DMA配置中断配置硬件流控制使能生成代码解析和使用方法串口初始化__UART_HandleTypeDef结构体浅析HAL库代码实际使用方法使用轮询方式发送使用轮询方式接收使用中断方式发送使用中…...
CSS | transition 和 transform的用处和区别
省流总结: transform用于变换/变形,transition是动画控制器 transform 用来对元素进行变形,常见的操作如下,它是立即生效的样式变形属性。 旋转 rotate(角度deg)、平移 translateX(像素px)、缩放 scale(倍数)、倾斜 skewX(角度…...
Go语言多线程问题
打印零与奇偶数(leetcode 1116) 方法1:使用互斥锁和条件变量 package mainimport ("fmt""sync" )type ZeroEvenOdd struct {n intzeroMutex sync.MutexevenMutex sync.MutexoddMutex sync.Mutexcurrent int…...
GitHub 趋势日报 (2025年06月06日)
📊 由 TrendForge 系统生成 | 🌐 https://trendforge.devlive.org/ 🌐 本日报中的项目描述已自动翻译为中文 📈 今日获星趋势图 今日获星趋势图 590 cognee 551 onlook 399 project-based-learning 348 build-your-own-x 320 ne…...

【JVM】Java虚拟机(二)——垃圾回收
目录 一、如何判断对象可以回收 (一)引用计数法 (二)可达性分析算法 二、垃圾回收算法 (一)标记清除 (二)标记整理 (三)复制 (四ÿ…...
快刀集(1): 一刀斩断视频片头广告
一刀流:用一个简单脚本,秒杀视频片头广告,还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农,平时写代码之余看看电影、补补片,是再正常不过的事。 电影嘛,要沉浸,…...

并发编程 - go版
1.并发编程基础概念 进程和线程 A. 进程是程序在操作系统中的一次执行过程,系统进行资源分配和调度的一个独立单位。B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位。C.一个进程可以创建和撤销多个线程;同一个进程中…...
Webpack性能优化:构建速度与体积优化策略
一、构建速度优化 1、升级Webpack和Node.js 优化效果:Webpack 4比Webpack 3构建时间降低60%-98%。原因: V8引擎优化(for of替代forEach、Map/Set替代Object)。默认使用更快的md4哈希算法。AST直接从Loa…...
Redis:现代应用开发的高效内存数据存储利器
一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发,其初衷是为了满足他自己的一个项目需求,即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源,Redis凭借其简单易用、…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...

免费数学几何作图web平台
光锐软件免费数学工具,maths,数学制图,数学作图,几何作图,几何,AR开发,AR教育,增强现实,软件公司,XR,MR,VR,虚拟仿真,虚拟现实,混合现实,教育科技产品,职业模拟培训,高保真VR场景,结构互动课件,元宇宙http://xaglare.c…...

基于Springboot+Vue的办公管理系统
角色: 管理员、员工 技术: 后端: SpringBoot, Vue2, MySQL, Mybatis-Plus 前端: Vue2, Element-UI, Axios, Echarts, Vue-Router 核心功能: 该办公管理系统是一个综合性的企业内部管理平台,旨在提升企业运营效率和员工管理水…...

Linux nano命令的基本使用
参考资料 GNU nanoを使いこなすnano基础 目录 一. 简介二. 文件打开2.1 普通方式打开文件2.2 只读方式打开文件 三. 文件查看3.1 打开文件时,显示行号3.2 翻页查看 四. 文件编辑4.1 Ctrl K 复制 和 Ctrl U 粘贴4.2 Alt/Esc U 撤回 五. 文件保存与退出5.1 Ctrl …...
Python+ZeroMQ实战:智能车辆状态监控与模拟模式自动切换
目录 关键点 技术实现1 技术实现2 摘要: 本文将介绍如何利用Python和ZeroMQ消息队列构建一个智能车辆状态监控系统。系统能够根据时间策略自动切换驾驶模式(自动驾驶、人工驾驶、远程驾驶、主动安全),并通过实时消息推送更新车…...

Razor编程中@Html的方法使用大全
文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...

Windows安装Miniconda
一、下载 https://www.anaconda.com/download/success 二、安装 三、配置镜像源 Anaconda/Miniconda pip 配置清华镜像源_anaconda配置清华源-CSDN博客 四、常用操作命令 Anaconda/Miniconda 基本操作命令_miniconda创建环境命令-CSDN博客...

iview框架主题色的应用
1.下载 less要使用3.0.0以下的版本 npm install less2.7.3 npm install less-loader4.0.52./src/config/theme.js文件 module.exports {yellow: {theme-color: #FDCE04},blue: {theme-color: #547CE7} }在sass中使用theme配置的颜色主题,无需引入,直接可…...
Web中间件--tomcat学习
Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机,它可以执行Java字节码。Java虚拟机是Java平台的一部分,Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...

Golang——6、指针和结构体
指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...

三分算法与DeepSeek辅助证明是单峰函数
前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...

宇树科技,改名了!
提到国内具身智能和机器人领域的代表企业,那宇树科技(Unitree)必须名列其榜。 最近,宇树科技的一项新变动消息在业界引发了不少关注和讨论,即: 宇树向其合作伙伴发布了一封公司名称变更函称,因…...
LRU 缓存机制详解与实现(Java版) + 力扣解决
📌 LRU 缓存机制详解与实现(Java版) 一、📖 问题背景 在日常开发中,我们经常会使用 缓存(Cache) 来提升性能。但由于内存有限,缓存不可能无限增长,于是需要策略决定&am…...

【网络安全】开源系统getshell漏洞挖掘
审计过程: 在入口文件admin/index.php中: 用户可以通过m,c,a等参数控制加载的文件和方法,在app/system/entrance.php中存在重点代码: 当M_TYPE system并且M_MODULE include时,会设置常量PATH_OWN_FILE为PATH_APP.M_T…...