0

0

C++shared_ptr和unique_ptr使用区别

P粉602998670

P粉602998670

发布时间:2025-09-17 13:00:02

|

660人浏览过

|

来源于php中文网

原创

unique_ptr独占所有权,性能高,适用于单一拥有者场景;shared_ptr共享所有权,通过引用计数管理生命周期,支持多拥有者但有性能开销和循环引用风险。

c++shared_ptr和unique_ptr使用区别

C++中的

shared_ptr
unique_ptr
,核心区别在于它们对资源所有权的管理哲学:
unique_ptr
奉行独占,而
shared_ptr
则支持共享。这不仅仅是语法上的差异,更深层次地触及到程序的性能、安全性和设计模式。简单来说,如果你需要一个资源只有一个明确的拥有者,并且在拥有者消失时资源也随之释放,那就选
unique_ptr
;如果你需要多个地方共同管理一个资源的生命周期,
shared_ptr
就是你的不二之选,但它也带来了额外的考量。

解决方案

unique_ptr
shared_ptr
是C++11引入的智能指针,旨在解决传统裸指针带来的内存泄漏和悬挂指针问题。它们的核心差异体现在所有权语义和底层实现上。

unique_ptr
代表的是独占所有权。这意味着在任何时刻,只有一个
unique_ptr
能够指向并管理一个特定的对象。这种所有权是严格的,不可复制,只能通过
std::move
进行转移。当一个
unique_ptr
超出其作用域或被销毁时,它所管理的对象也会被自动删除。它的优势在于零运行时开销(除了对象本身的析构),因为它不需要维护引用计数,所有的所有权检查都是在编译期完成的,性能上与裸指针几乎无异。

shared_ptr
则实现了共享所有权。多个
shared_ptr
可以共同指向并管理同一个对象。它通过内部维护一个引用计数器来实现这一点:每当一个新的
shared_ptr
指向该对象时,引用计数器加一;每当一个
shared_ptr
被销毁或不再指向该对象时,引用计数器减一。只有当引用计数器归零时,
shared_ptr
才会负责删除所管理的对象。这种机制极大地简化了复杂对象图的生命周期管理,但代价是需要额外的内存来存储引用计数,并且引用计数的增减操作(通常是原子操作)会带来一定的运行时开销。

立即学习C++免费学习笔记(深入)”;

选择哪种智能指针,关键在于你对资源所有权的需求。如果资源有一个明确的、唯一的拥有者,并且这个拥有者在其生命周期结束时负责释放资源,那么

unique_ptr
无疑是更优的选择,因为它提供了更好的性能和编译期安全性。如果资源需要在程序的多个部分之间共享,并且没有一个单一的拥有者,那么
shared_ptr
的共享所有权模型就能很好地解决问题,尽管你需要注意它可能带来的性能和循环引用问题。

std::unique_ptr
的性能优势与适用场景

在我看来,

unique_ptr
是C++智能指针家族中的“性能之王”。它的设计哲学就是最小化开销,提供与裸指针相近的性能表现,同时又保证了内存安全。这种性能优势主要来源于其独占所有权模型,这意味着它根本不需要像
shared_ptr
那样去维护一个引用计数。没有引用计数的增减,就没有原子操作的开销,也没有额外的内存分配来存储这个计数器。说白了,它就是一个带RAII(资源获取即初始化)语义的裸指针,在编译期就锁定了资源的唯一归属。

它的适用场景非常广泛,几乎涵盖了所有你可以明确资源单一所有权的情况。比如,当你在一个工厂函数中创建对象并返回时,

unique_ptr
是理想的选择:

std::unique_ptr createObject() {
    return std::make_unique(); // 返回一个独占所有权的智能指针
}
// ...
auto obj = createObject(); // obj现在独占MyObject实例

再比如,在一个类中,如果某个成员变量是动态分配的,并且它的生命周期完全由这个类的实例来管理,那么

unique_ptr
是比裸指针更好的选择,它能确保当类的实例被销毁时,动态分配的成员也会被正确释放,避免了手动
delete
的繁琐和遗漏。

class Manager {
private:
    std::unique_ptr _resource; // Manager独占Resource
public:
    Manager() : _resource(std::make_unique()) {}
    // ...
};

此外,在标准库容器中存储动态分配的对象时,

std::vector>
是一个非常常见的模式。它允许你存储大量独立的对象,并且在容器销毁时,所有这些对象都会被自动清理,效率很高。这些场景都体现了
unique_ptr
在保证安全性的同时,对性能的极致追求。

网钛淘拍CMS(TaoPaiCMS) V1.60
网钛淘拍CMS(TaoPaiCMS) V1.60

2013年07月06日 V1.60 升级包更新方式:admin文件夹改成你后台目录名,然后补丁包里的所有文件覆盖进去。1.[新增]后台引导页加入非IE浏览器提示,后台部分功能在非IE浏览器下可能没法使用2.[改进]淘客商品管理 首页 列表页 内容页 的下拉项加入颜色来区别不同项3.[改进]后台新增/修改淘客商品,增加淘宝字样的图标和天猫字样图标改成天猫logo图标4.[改进]为统一名称,“分类”改

下载

std::shared_ptr
的生命周期管理与陷阱

shared_ptr
的出现,确实让C++中复杂对象图的生命周期管理变得前所未有的简单。它的核心机制是引用计数:每当一个
shared_ptr
被复制,指向同一个资源,内部的引用计数就会增加;当一个
shared_ptr
被销毁或重置,引用计数就会减少。只有当引用计数降到零时,它所管理的对象才会被真正释放。这种“多人共管”的模式,在很多场景下都非常方便,比如当一个对象需要被多个模块、多个线程同时访问,并且这些模块的生命周期不完全同步时。

然而,这种便利并非没有代价,它最臭名昭著的陷阱就是循环引用。想象一下,两个对象A和B,A持有一个

shared_ptr
指向B,B也持有一个
shared_ptr
指向A。当这两个对象被创建后,它们的引用计数都会是2(自身一个,对方一个)。即便外部所有指向A和B的
shared_ptr
都消失了,A和B内部的
shared_ptr
依然存在,导致它们的引用计数永远不会降到零。结果就是,这两个对象会永远驻留在内存中,造成内存泄漏。

struct B; // 前向声明
struct A {
    std::shared_ptr b_ptr;
    ~A() { std::cout << "A destroyed!\n"; }
};

struct B {
    std::shared_ptr a_ptr; // 这里如果用shared_ptr,就会形成循环引用
    ~B() { std::cout << "B destroyed!\n"; }
};

void create_circular_ref() {
    auto a = std::make_shared();
    auto b = std::make_shared();
    a->b_ptr = b;
    b->a_ptr = a;
} // 当a和b离开作用域时,它们的引用计数仍为1,导致A和B都不会被销毁

为了解决这个难题,C++标准库引入了

std::weak_ptr
weak_ptr
是一种不拥有资源所有权的智能指针,它指向一个由
shared_ptr
管理的对象,但不会增加该对象的引用计数。你可以把它看作是一个“观察者”,它能检查对象是否还存在,但不会阻止对象的销毁。当需要访问
weak_ptr
指向的对象时,可以通过其
lock()
方法尝试获取一个
shared_ptr
。如果对象已经销毁,
lock()
会返回一个空的
shared_ptr

// 修正后的B结构体,使用weak_ptr打破循环引用
struct B_fixed {
    std::weak_ptr a_ptr; // 使用weak_ptr
    ~B_fixed() { std::cout << "B_fixed destroyed!\n"; }
};

void create_no_circular_ref() {
    auto a = std::make_shared();
    auto b = std::make_shared();
    a->b_ptr = b;
    b->a_ptr = a; // 这里a_ptr不会增加a的引用计数
} // 当a和b离开作用域时,A和B_fixed都会被正确销毁

所以,在使用

shared_ptr
时,尤其是在设计相互引用的对象时,务必审视是否存在循环引用的可能。
weak_ptr
是解决这个问题的关键
工具,它允许你建立非所有权关系,从而正确管理对象的生命周期。

智能指针间的转换与协作:
unique_ptr
shared_ptr

在实际的软件开发中,我们经常会遇到这样的场景:一个资源最初被设计为独占所有权,由

unique_ptr
管理,但在程序的某个阶段,这个资源突然需要被多个模块共享。这时,我们就需要将
unique_ptr
“升级”为
shared_ptr
。这种转换是完全合理且被C++标准支持的。

unique_ptr
shared_ptr
的转换,本质上是所有权语义的转变:从独占变为共享。这个过程通常是通过
std::move
来实现的,因为
unique_ptr
是不可复制的。

std::unique_ptr unique_gadget = std::make_unique();
// ... 经过一些独占阶段的操作 ...

// 现在,这个Gadget需要被共享了
std::shared_ptr shared_gadget = std::move(unique_gadget);
// 此时,unique_gadget已经变为空指针,所有权已转移给shared_gadget
// shared_gadget的引用计数为1

这种转换是单向的。你不能将一个

shared_ptr
直接转换为
unique_ptr
,因为
shared_ptr
可能已经有多个拥有者,强行转换为
unique_ptr
会破坏其共享所有权的语义,导致其他
shared_ptr
变成悬挂指针。如果你真的需要从
shared_ptr
中获取一个裸指针,并打算对其进行独占管理(这通常是危险的,需要非常小心),你必须确保那是最后一个
shared_ptr
,并且你手动接管了资源的生命周期,这通常不推荐。

这种转换机制体现了C++智能指针的灵活性和实用性。它允许开发者根据程序运行时的实际需求,动态调整资源的生命周期管理策略。在一个大型系统中,一个对象可能在某个阶段作为内部组件被独占管理,而在另一个阶段又需要作为API的一部分被广泛共享。

unique_ptr
shared_ptr
的转换,为这种灵活的设计提供了强有力的支持,让我们的代码能够更好地适应不断变化的需求。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

525

2023.08.10

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

278

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

213

2023.12.29

2026赚钱平台入口大全
2026赚钱平台入口大全

2026年最新赚钱平台入口汇总,涵盖任务众包、内容创作、电商运营、技能变现等多类正规渠道,助你轻松开启副业增收之路。阅读专题下面的文章了解更多详细内容。

52

2026.01.31

高干文在线阅读网站大全
高干文在线阅读网站大全

汇集热门1v1高干文免费阅读资源,涵盖都市言情、京味大院、军旅高干等经典题材,情节紧凑、人物鲜明。阅读专题下面的文章了解更多详细内容。

40

2026.01.31

无需付费的漫画app大全
无需付费的漫画app大全

想找真正免费又无套路的漫画App?本合集精选多款永久免费、资源丰富、无广告干扰的优质漫画应用,涵盖国漫、日漫、韩漫及经典老番,满足各类阅读需求。阅读专题下面的文章了解更多详细内容。

50

2026.01.31

漫画免费在线观看地址大全
漫画免费在线观看地址大全

想找免费又资源丰富的漫画网站?本合集精选2025-2026年热门平台,涵盖国漫、日漫、韩漫等多类型作品,支持高清流畅阅读与离线缓存。阅读专题下面的文章了解更多详细内容。

11

2026.01.31

漫画防走失登陆入口大全
漫画防走失登陆入口大全

2026最新漫画防走失登录入口合集,汇总多个稳定可用网址,助你畅享高清无广告漫画阅读体验。阅读专题下面的文章了解更多详细内容。

13

2026.01.31

php多线程怎么实现
php多线程怎么实现

PHP本身不支持原生多线程,但可通过扩展如pthreads、Swoole或结合多进程、协程等方式实现并发处理。阅读专题下面的文章了解更多详细内容。

1

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.4万人学习

Pandas 教程
Pandas 教程

共15课时 | 1.0万人学习

ASP 教程
ASP 教程

共34课时 | 4.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号