0

0

C++里的智能指针unique_ptr有什么特点?(独占所有权与不可拷贝)

穿越時空

穿越時空

发布时间:2026-01-18 16:02:02

|

486人浏览过

|

来源于php中文网

原创

unique_ptr不能拷贝,因其构造函数和赋值运算符被显式删除,以确保资源独占;拷贝会导致双重释放,故仅支持通过std::move转移所有权,移动后原指针为空。

c++里的智能指针unique_ptr有什么特点?(独占所有权与不可拷贝)

unique_ptr 为什么不能拷贝?

因为 unique_ptr 的设计目标就是「独占资源」,它的构造函数和赋值运算符都被显式删除了(= delete)。一旦允许拷贝,两个指针就可能同时指向同一块内存,析构时触发双重释放——这是未定义行为,程序大概率崩溃。

常见错误现象:

  • unique_ptr p1 = make_unique(42); unique_ptr p2 = p1; → 编译失败,报错类似 use of deleted function 'std::unique_ptr<_tp _dp>::unique_ptr(const std::unique_ptr<_tp _dp>&)'
  • 传参时写 void func(unique_ptr p) 然后调用 func(p1) → 同样编译失败,除非你主动用 std::move(p1)

移动语义是唯一合法的“转移”方式

所有权只能通过移动(std::move)转移,移动后原指针变为空(get() == nullptr),新指针接管资源。这符合「独占」的语义约束,也避免了运行时开销。

实操要点:

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

Designs.ai
Designs.ai

AI设计工具

下载
  • 移动后原 unique_ptr 不再持有对象,再次解引用(如 *p1)会触发空指针解引用,行为未定义
  • 函数返回 unique_ptr 时,现代编译器通常启用 RVO 或移动优化,无需手动 std::move;但函数参数接收时必须显式 std::move
  • 容器中存储 unique_ptr(如 vector>)是安全且高效的,插入/删除都靠移动完成
unique_ptr create_int() {
    return make_unique(100); // OK:返回值自动移动
}
unique_ptr p1 = create_int(); // OK

unique_ptr p2 = std::move(p1); // OK:显式转移 // cout << p1; // 危险!p1 已为空 cout << p2; // 输出 100

和 raw pointer、shared_ptr 的关键区别在哪?

对比不是为了教你怎么选,而是帮你判断「是不是真该用 unique_ptr」:

  • vs raw pointer:unique_ptr 自动析构,不用手写 delete;但不能隐式转成 T*,需调用 .get()(仅获取地址,不移交所有权)
  • vs shared_ptrunique_ptr 零开销(无引用计数)、无线程同步成本;但无法共享所有权——如果多个模块需要同时访问同一对象,它就不合适
  • 适用场景典型包括:函数内部临时资源管理、PIMPL 惯用法、工厂函数返回、STL 容器元素(替代裸指针数组)

容易被忽略的细节:自定义删除器与数组特化

默认情况下 unique_ptrdelete,但若你管理的是数组(new T[10])或 C 风格资源(如 fopen 文件句柄),必须显式指定删除器,否则行为未定义。

  • 数组用法必须写成 unique_ptr,否则 delete 会调用错的析构路径
  • 自定义删除器会改变类型: unique_ptrunique_ptr 是不同类型,不能互相赋值(即使都用 std::move
  • 删除器对象本身会被存储在 unique_ptr 实例里,若删除器是函数指针(轻量),几乎无额外开销;若为带状态的 lambda,则需注意大小
// 正确的数组用法
unique_ptr arr(new int[5]{1,2,3,4,5});

// 正确的 FILE 管理 auto del_file = [](FILE f) { if (f) fclose(f); }; unique_ptr fp(fopen("a.txt", "r"), del_file);

真正用好 unique_ptr 的难点不在语法,而在于时刻意识到:它代表一种明确的所有权契约——谁 move 谁负责,move 后原变量即失效,没有商量余地。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1465

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

204

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

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

精品课程

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

共94课时 | 7万人学习

C 教程
C 教程

共75课时 | 4.1万人学习

C++教程
C++教程

共115课时 | 12.7万人学习

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

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