0

0

c++怎么实现深拷贝与浅拷贝_c++ 拷贝构造函数与赋值运算符重载【方法】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-04 18:53:02

|

965人浏览过

|

来源于php中文网

原创

浅拷贝是编译器默认的逐字节复制,对指针只复制地址,易导致double free、数据误改等问题;深拷贝需手动实现拷贝构造函数和赋值运算符,为动态资源分配新内存并复制内容,同时处理自赋值、异常安全等细节。

c++怎么实现深拷贝与浅拷贝_c++ 拷贝构造函数与赋值运算符重载【方法】

浅拷贝就是默认的,但默认行为往往不是你想要的

当你没写拷贝构造函数或 operator= 时,C++ 编译器会自动生成一个——它逐字节复制对象内存,对指针成员只复制地址,不复制所指内容。这叫浅拷贝。

问题就出在:如果类里有 new 出来的堆内存(比如 int*std::string 内部指针等),两个对象会指向同一块内存。析构时两次 delete 同一地址,直接触发 double free or corruption 错误。

常见错误现象:

  • 程序运行到第二次析构时崩溃(SIGABRT)
  • 修改对象 A 的数据,对象 B 的对应字段也变了
  • Valgrind 报告 Invalid read/writeUse after free

深拷贝必须手动实现拷贝构造函数和赋值运算符

核心原则:为每个动态分配的资源单独申请新内存,并把原内容完整复制过去。不能依赖编译器生成的默认版本。

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

实操要点:

Baklib
Baklib

在线创建产品手册、知识库、帮助文档

下载
  • 拷贝构造函数签名必须是 A(const A& other),且不能是 const A& 以外的引用类型
  • operator= 必须返回 A&,并处理自赋值:if (this == &other) return *this;
  • 赋值前要先释放当前对象已持有的资源(避免内存泄漏)
  • 两者逻辑高度重复,可提取共用的 copy_from(const A& other) 辅助函数
class String {
private:
    char* data_;
    size_t len_;

public: String(const char* s) : len(s ? strlen(s) : 0) { data = new char[len + 1]; if (s) strcpy(data, s); else data_[0] = '\0'; }

// 深拷贝构造函数
String(const String& other) : len_(other.len_) {
    data_ = new char[len_ + 1];
    strcpy(data_, other.data_);
}

// 深拷贝赋值运算符
String& operator=(const String& other) {
    if (this == &other) return *this;
    delete[] data_;  // 先释放旧资源
    len_ = other.len_;
    data_ = new char[len_ + 1];
    strcpy(data_, other.data_);
    return *this;
}

~String() { delete[] data_; }

};

现代 C++ 推荐用 RAII 和移动语义替代手写深拷贝

手动管理 new/delete 容易出错。C++11 起,优先用标准容器封装资源:

  • char* 换成 std::string,把 int* 换成 std::vector
  • 这些类型自身已正确实现深拷贝,你无需再写拷贝构造函数或 operator=
  • 若仍需自定义资源管理,应继承 RAII 原则:构造获取资源,析构释放资源,拷贝/移动按需实现
  • C++11 后,operator= 还应支持移动赋值(A& operator=(A&& other)),避免不必要的深拷贝开销

性能影响明显:对大对象(如百万级 std::vector),深拷贝是 O(n) 时间,而移动是 O(1);忽略移动语义会导致隐式深拷贝,尤其在 std::vector 扩容时放大问题。

容易被忽略的细节:const 成员、mutable、基类与虚函数

深拷贝逻辑看似简单,但几个边界情况常被跳过:

  • const 成员变量不能在构造函数体里赋值,必须在成员初始化列表中完成深拷贝(如 : const_ptr_(new int(*other.const_ptr_))
  • mutable 成员虽可修改,但若它内部含指针,同样需要深拷贝(例如缓存用的 mutable std::unique_ptr
  • 派生类实现深拷贝时,必须显式调用基类的拷贝构造函数:Derived(const Derived& d) : Base(d), ...,否则基类部分仍是浅拷贝
  • 含虚函数的类,若拷贝后需保持多态性,确保虚表指针正常继承;一般只要基类拷贝正确,这点由编译器保障

最常漏的是自赋值检查和异常安全:如果 new 抛出 std::bad_alloc,原对象状态可能已被破坏。更健壮的做法是“拷贝-交换”惯用法(copy-and-swap),但前提是你的类支持 swap 且无异常。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

314

2023.08.02

java基础知识汇总
java基础知识汇总

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

1458

2023.10.24

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

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

227

2024.02.23

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

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

85

2025.10.17

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

719

2023.08.22

java多态详细介绍
java多态详细介绍

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

15

2025.11.27

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

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

520

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

314

2023.08.02

免费看漫画app合集_2026免费漫画app排行榜入口
免费看漫画app合集_2026免费漫画app排行榜入口

2026年免费漫画APP合集来啦!为你精心整理最新免费漫画APP排行榜入口,涵盖漫蛙漫画、香香漫画、包子漫画等热门神器,海量正版国漫、日漫、韩漫资源全免费阅读,无需付费解锁章节!高清全彩画质、每日极速更新,支持离线下载、智能推荐、条漫阅读模式,热血、恋爱、悬疑、古风、搞笑等题材应有尽有。无论你是追新番老粉还是小白漫迷,这里都能让你一站式追漫到爽,告别广告干扰和会员套路!赶紧点击入口下载体验,开启2026无限免费漫画之旅吧!

14

2026.01.07

热门下载

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

精品课程

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

共18课时 | 4.3万人学习

Sass 教程
Sass 教程

共14课时 | 0.7万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

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

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