0

0

c++ 拷贝构造函数写法 c++何时需要深拷贝和浅拷贝

幻夢星雲

幻夢星雲

发布时间:2026-03-16 21:27:12

|

104人浏览过

|

来源于php中文网

原创

正确写法是参数为const T&、初始化列表中初始化所有成员、函数体内手动深拷贝资源且不调用operator=;必须深拷贝的场景包括裸指针指向堆内存、持有文件/socket句柄等独占资源。

c++ 拷贝构造函数写法 c++何时需要深拷贝和浅拷贝

拷贝构造函数怎么写才不会出错

默认生成的拷贝构造函数只做成员逐个赋值(即浅拷贝),只要类里没指针、没动态分配内存、没文件句柄或类似资源,它就足够安全。但一旦有 int*std::vector 以外的自定义资源管理逻辑,就必须自己写。

正确写法核心就三点:参数必须是 const T&,函数体里手动复制所有资源,且不能调用 operator=(那是赋值,不是构造):

class Buffer {
    char* data_;
    size_t size_;
public:
    Buffer(const Buffer& other) : data_(nullptr), size_(other.size_) {
        if (other.data_) {
            data_ = new char[size_];
            std::memcpy(data_, other.data_, size_);
        }
    }
};
  • 漏掉初始化列表里的 data_(nullptr),后续 if 判断可能读未初始化内存
  • 把参数写成 Buffer other(值传递)会无限递归调用拷贝构造函数
  • 在函数体内用 *this = other; 是错的——这是赋值操作,不是构造,而且可能触发未定义行为

哪些场景必须深拷贝

深拷贝的本质是「每个对象独占一份资源」。只要两个对象共享同一块堆内存、同一个文件描述符、同一个 socket 句柄,就存在资源竞争或提前释放风险。

  • char*void* 指向堆内存(比如用 newmalloc 分配的)
  • 持有 FILE*int fd(Linux 文件描述符)、HANDLE(Windows)等系统资源
  • 内部缓存了某个全局结构体的引用或指针,且该结构体生命周期不由当前类控制
  • 使用了 std::shared_ptr 以外的裸指针管理对象图(比如树节点之间用 Node* 连接)

反例:用 std::stringstd::vectorstd::shared_ptr 管理资源时,通常不用手写深拷贝——它们自己已经实现了正确的拷贝语义。

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

会译·对照式翻译
会译·对照式翻译

会译是一款AI智能翻译浏览器插件,支持多语种对照式翻译

下载

浅拷贝不是 bug,但容易被误用

浅拷贝本身完全合法,C++ 默认就是它。问题出在「你以为它安全,其实不安全」。常见错误现象包括:

  • 程序运行一阵后崩溃,报 double free or corruption —— 两个对象析构时都 delete 同一块内存
  • 一个对象修改了数据,另一个对象的值也变了 —— 它们指向同一片内存,但本意是各自独立
  • 移动后原对象仍被使用,而你没禁用拷贝(比如没把拷贝构造函数设为 = delete

注意:即使写了深拷贝,也要同步检查 operator= 和析构函数,三者必须语义一致(Rule of Three)。C++11 起更推荐直接用 = default= delete 显式声明,避免隐式生成带来的歧义。

现代 C++ 里怎么绕过手写深拷贝

不是所有情况都得自己 new/delete。多数时候,改用 RAII 容器就能让编译器继续用默认拷贝构造函数,且语义正确。

  • char* data_ 换成 std::vector<char> data_,拷贝自动深拷
  • FILE* 封装进自定义 RAII 类(如 FileHandle),并在其拷贝构造函数里执行 fopen + fseek 复制内容,而非共享句柄
  • 明确不需要拷贝?直接删掉:Buffer(const Buffer&) = delete;
  • 只允许移动?写 Buffer(Buffer&&) = default; 并删掉拷贝(或反过来)

最易被忽略的一点:深拷贝成本可能很高(比如几百 MB 的 buffer),而你根本没意识到——因为没测过拷贝耗时,也没加日志。上线后突然发现某次 vector resize 触发大量对象拷贝,CPU 直接拉满。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1071

2023.08.02

if什么意思
if什么意思

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

847

2023.08.22

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

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

565

2023.09.20

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

510

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

204

2025.07.04

string转int
string转int

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

1071

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

617

2024.08.29

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

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

335

2025.08.29

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

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

精品课程

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

共94课时 | 11.5万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22.2万人学习

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

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