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

C++ 常用的智能指针

C++ 智能指针


一、智能指针类型概览

C++ 标准库提供以下智能指针(需包含头文件 <memory>):

  1. unique_ptr:独占所有权,不可复制, 可移动
  2. shared_ptr:共享所有权,用于引用计数
  3. weak_ptr:观察 shared_ptr 对象,不增加计数

二、unique_ptr(独占指针)

核心特性

  • 唯一拥有指向对象的所有权
  • 支持移动语义(所有权转移)
  • 离开作用域自动释放内存
#include <iostream>
#include <memory>void unique_ptr_demo() {// 创建智能指针 (C++14推荐make_unique)std::unique_ptr<int> p1 = std::make_unique<int>(10);// 访问数据std::cout << *p1 << std::endl;  // 输出:10// 所有权转移std::unique_ptr<int> p2 = std::move(p1);std::cout << (p1 ? "非空" : "空") << std::endl;  // 输出:空
}int main() {unique_ptr_demo();return 0;
}

三、shared_ptr(共享指针)

核心特性

  • 通过引用计数共享所有权
  • 最后一个指针销毁时释放内存
  • 支持自定义删除器
#include <iostream>
#include <memory>class Data {
public:Data() { std::cout << "Data构造" << std::endl; }~Data() { std::cout << "Data析构" << std::endl; }
};void shared_ptr_demo() {std::shared_ptr<Data> p1;{ // 内部作用域auto p2 = std::make_shared<Data>();p1 = p2;  // 共享所有权std::cout << "引用计数: " << p1.use_count() << std::endl;  // 输出:2}std::cout << "引用计数: " << p1.use_count() << std::endl;  // 输出:1
}int main() {shared_ptr_demo();return 0;
}

share_ptr引起的循环引用的问题。循环引用就是内存泄漏。 因为对象之间相互引用,使得引用计数永远不会变为 0 ,对象所占用的堆内存一直无法被释放。随着程序运行,不断出现这种循环引用的情况,会导致可用内存越来越少,最终可能使程序崩溃,或者引发系统性能问题。

#include <memory>
#include <iostream>class B;
class A {
public:std::shared_ptr<B> bptr;~A() {std::cout << "~A()" << endl;}
};class B {
public:std::shared_ptr<A> aptr;~B() {std::cout << "~B()" << endl;}
};int main() {std::shared_ptr<A> aa = std::make_shared<A>();std::shared_ptr<B> bb = std::make_shared<B>();aa->bptr = bb;bb->aptr = aa;return 0;
}

四、weak_ptr(弱指针)

weak_ptr 是一种弱引用智能指针,它指向由 shared_ptr 管理的对象,但不增加对象的引用计数
核心特性

  • 不增加引用计数
  • 解决 shared_ptr 循环引用问题
  • 必须转换为 shared_ptr 才能访问数据
#include <iostream>
#include <memory>class Node {
public:std::weak_ptr<Node> partner;  // 使用weak_ptr打破循环~Node() { std::cout << "节点销毁" << std::endl; }
};void weak_ptr_demo() {auto node1 = std::make_shared<Node>();auto node2 = std::make_shared<Node>();node1->partner = node2;node2->partner = node1;if(auto sp = node1->partner.lock()) { std::cout << "有效伙伴节点" << std::endl; }
}int main() {weak_ptr_demo();return 0;
}

五、编译与运行
  1. 编译命令

    g++ -std=c++14 -o smart_ptr_demo smart_ptr_demo.cpp
    
  2. 运行结果

    Data构造
    引用计数: 2
    引用计数: 1
    Data析构
    节点销毁
    节点销毁
    

六、选择指南
指针类型使用场景性能开销
unique_ptr独占资源的场景(90%日常使用)
shared_ptr需要共享所有权的复杂场景
weak_ptr解决循环引用或观察共享资源

最佳实践

  1. 优先使用 make_unique/make_shared 创建指针
  2. 避免裸指针与智能指针混用
  3. 明确所有权关系,能用 unique_ptr 就不用 shared_ptr
http://www.xdnf.cn/news/17281.html

相关文章:

  • 使用Docker搭建开源Email服务器
  • 高防IP如何针对DDoS攻击特点起防护作用
  • 小刚说C语言刷题——1033 判断奇偶数
  • 《GPT-4.1深度解析:AI进化新标杆,如何重塑行业未来?》
  • Spring数据访问全解析:ORM整合与JDBC高效实践
  • 【Mysql】mysql数据库占用空间查询
  • 基础编程题目集 6-2 多项式求值
  • VUE简介
  • 蓝桥杯12. 日期问题
  • 全面解析IPv6:从理论到实践(以H3C配置为例)
  • 搜索插入位置--LeetCode
  • Linux中find和grep的区别
  • 常见但是有挑战的效果组件鸿蒙版
  • 视频分析设备平台EasyCVR化解高速服务区管理难题,打造全方位智能安防监控方案
  • 第 5 期(进阶版):训练第一个 DDPM 模型(使用 CIFAR-10 数据集)
  • 服务器上有conda环境 退出conda环境 再安装uv包管理器这样子就不会有冲突吗
  • MQ基础篇
  • LoRA(Low - Rank Adaptation,低秩自适应 )微调技术改进创新点
  • 并发设计模式实战系列(1):半同步/半异步模式
  • day45——非递减数列(LeetCode-665)
  • QT项目打包
  • Multi Agents Collaboration OS:文档合规性及质量检测助手设计及实践
  • 【KWDB 创作者计划】_算法篇---Stockwell变换
  • OpenAI重返巅峰:o3与o4-mini引领AI推理新时代
  • 面试经验杂谈
  • onlyoffice关闭JWT后依然报错如何解决?
  • Maven 简介(图文)
  • 文件系统的npu和内核的npu有什么区别
  • 双层Key缓存
  • 【dify实战】agent结合deepseek实现基于自然语言的数据库问答、Echarts可视化展示、Excel报表下载