0

0

C++如何实现对象的深拷贝克隆接口?(纯虚clone方法)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-19 12:02:39

|

596人浏览过

|

来源于php中文网

原创

必须返回 std::unique_ptr 或 base* 而非 base 值类型,否则发生对象切片,丢失派生类型信息与多态行为;推荐 std::unique_ptr 以明确所有权、自动释放并支持协变返回类型。

c++如何实现对象的深拷贝克隆接口?(纯虚clone方法)

为什么 clone() 必须返回 std::unique_ptr<base>Base*,而不是 Base 值类型?

因为深拷贝的目标是复制出一个新对象,且要保留其原始动态类型(比如 Derived 实例克隆后仍是 Derived,不是切片成 Base)。如果返回 Base 值,会发生对象切片 —— 派生部分被丢弃,拷贝失去意义。

  • 返回 Base 值 → 编译可能通过,但运行时丢失派生状态,dynamic_cast 失败,虚函数调用也退化为 Base 版本
  • 返回 Base* → 调用方必须手动 delete,容易泄漏;且无法表达“所有权转移”语义
  • 推荐返回 std::unique_ptr<base> → 明确所有权,自动释放,支持多态,现代 C++ 标准做法

纯虚 clone() 接口怎么写才不踩坑?

接口声明看似简单,但几个细节错一点就编译失败或行为异常:

  • 必须是 const 成员函数:克隆不应改变原对象状态
  • 返回类型需与派生类重写的返回类型协变(C++11 起支持)——基类写 virtual std::unique_ptr<base> clone() const = 0;,派生类可写 std::unique_ptr<derived> clone() const override;</derived>
  • 不能漏掉 override:否则重写失败(比如拼错函数名、参数列表不一致)时编译器不会报错,而是悄悄变成新函数,导致多态失效
  • 避免在基类提供默认实现:除非你确定所有派生类都能统一用 new Derived(*this),但多数场景下派生类构造逻辑各异,强制由子类实现更安全

clone() 实现里 new 出的对象,为什么建议用 std::make_unique 而不是裸 new

new 容易引发异常安全问题:如果构造函数抛异常,new 分配的内存不会自动释放;而 std::make_unique 是异常安全的,且语义更清晰。

错误写法:return std::unique_ptr<derived>(new Derived(*this));</derived>

SoundRaw AI
SoundRaw AI

面向创作者的 AI 音乐生成器,只需选择情绪、流派和长度,SoundRaw AI就能为你生成优美的歌曲。

下载

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

正确写法:return std::make_unique<derived>(*this);</derived>

  • std::make_unique 保证构造和分配原子性,不会内存泄漏
  • 不需要显式写类型两次(<derived></derived> 只出现一次),减少模板冗余
  • 如果 Derived 构造函数是 explicit 或有多个重载,make_unique 仍能正常推导和转发参数

继承链深了以后,clone() 怎么避免重复写样板代码?

三层以上继承(Base ← Intermediate ← Final)时,每个中间类都得重写 clone(),稍不注意就会写成 return std::make_unique<intermediate>(*this);</intermediate> —— 这会切断多态,Final 对象克隆后变成 Intermediate

  • 根本原则:每个类只负责“自己这一层”的拷贝,不越级假设子类存在
  • 中间类(如 Intermediate)的 clone() 必须仍返回 std::unique_ptr<intermediate></intermediate>,且内部调用 new Intermediate(*this)(或 make_unique),不能试图 downcast
  • 真正需要完整克隆的,是叶子类(Final);它知道自己完整结构,才能安全构造自身副本
  • 如果真想减少样板,可用 CRTP 辅助,但代价是侵入性和可读性下降 —— 多数项目里老老实实重写更稳妥

最常被忽略的一点:clone() 的语义是“深拷贝”,但如果你的类里含有裸指针、FILE*、系统句柄等非 RAII 资源,*this 拷贝构造本身不会处理它们 —— 这些必须在拷贝构造函数里显式深拷贝或重新获取,clone() 只是触发了它。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

24

2025.11.27

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

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

547

2023.09.20

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1529

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

423

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2260

2025.12.29

java接口相关教程
java接口相关教程

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

37

2026.01.19

go语言 数组和切片
go语言 数组和切片

本专题整合了go语言数组和切片的区别与含义,阅读专题下面的文章了解更多详细内容。

47

2025.09.03

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

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

287

2023.11.13

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

660

2026.02.13

热门下载

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

精品课程

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

共94课时 | 9.7万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18.3万人学习

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

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