0

0

C++深拷贝和浅拷贝区别 C++拷贝构造函数编写注意事项【必看】

尼克

尼克

发布时间:2026-02-01 12:45:30

|

428人浏览过

|

来源于php中文网

原创

浅拷贝是位拷贝,仅复制指针值而非内容,易致double free;深拷贝需自定义拷贝构造函数(A(const A&))和赋值运算符,用拷贝-交换法确保异常安全与自赋值防护,并遵循五法则实现资源管理。

c++深拷贝和浅拷贝区别 c++拷贝构造函数编写注意事项【必看】

浅拷贝就是位拷贝,深拷贝要自己写

默认的拷贝构造函数和赋值运算符只做浅拷贝:它逐字节复制对象内存,对指针成员来说,只是复制了指针值(地址),而不是指针指向的内容。如果原对象析构时 delete 了那块堆内存,副本再访问或析构就会触发 double freeuse-after-free——这是最典型的崩溃根源。

深拷贝必须手动分配新内存、逐个复制数据。核心判断标准就一条:只要类里有指针成员(尤其是指向堆内存的)或资源句柄(如文件描述符、socket),就必须自定义拷贝构造函数和 operator=

拷贝构造函数里别漏掉 const 引用参数

合法签名只能是 A(const A& other)。写成 A(A& other) 会导致无法接受临时对象(比如函数返回值),写成 A(A other) 会无限递归调用自身——因为传参会再次触发拷贝构造。

常见错误示例:

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

A(A other) {  // ❌ 错误:传值导致无限递归
    data = other.data;
}

正确写法必须带 const &

A(const A& other) : data(nullptr) {
    if (other.data) {
        data = new int[other.size];
        std::copy(other.data, other.data + other.size, data);
        size = other.size;
    }
}

赋值运算符要处理自赋值和异常安全

浅拷贝的默认 operator= 同样只复制指针,而且不检查自赋值(a = a)。直接 delete data; data = other.data; 会导致 data 被删后又去读已释放内存。

GoEnhance
GoEnhance

全能AI视频制作平台:通过GoEnhance AI让视频创作变得比以往任何时候都更简单。

下载

推荐用“拷贝-交换”惯用法(copy-and-swap),天然规避自赋值和部分异常风险:

A& operator=(A other) {  // 注意:这里传值,触发拷贝构造
    swap(*this, other);
    return *this;
}
void swap(A& a, A& b) noexcept {
    std::swap(a.data, b.data);
    std::swap(a.size, b.size);
}

关键点:

  • 参数用值传递,让编译器自动调用你的拷贝构造函数
  • swap 必须是 noexcept,否则异常时可能破坏强异常保证
  • 别在 operator= 里手动 deletenew,容易出错

移动语义出现后,深/浅拷贝逻辑依然不能省

C++11 后有了移动构造函数和移动赋值,但它们不会替代拷贝逻辑——只是提供更高效的资源转移方式。如果你没声明移动操作,编译器可能生成默认版本(对指针成员仍是浅移动),也可能因存在用户定义的拷贝操作而抑制移动函数生成。

真正安全的做法是:显式定义全部五个特殊成员函数(俗称“五法则”):

  • A(const A&)(深拷贝)
  • A& operator=(const A&)(深赋值)
  • A(A&&)(移动构造,通常把指针置为 nullptr
  • A& operator=(A&&)(移动赋值)
  • ~A()(确保 delete 堆内存)

最容易被忽略的是:即使你写了移动操作,拷贝构造函数仍需保持正确——因为有些场景(如容器扩容时的元素复制)仍会调用它。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

1503

2023.10.24

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

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

233

2024.02.23

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

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

87

2025.10.17

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

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

532

2023.09.20

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

113

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

103

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

399

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

575

2023.08.10

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

30

2026.01.31

热门下载

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

精品课程

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

共18课时 | 5.1万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 1万人学习

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

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