0

0

怎样编写C++中的拷贝控制成员 三五法则实践指南

P粉602998670

P粉602998670

发布时间:2025-07-21 08:40:02

|

342人浏览过

|

来源于php中文网

原创

三五法则指当类需要自定义析构函数、拷贝构造函数、拷贝赋值运算符、移动构造函数或移动赋值运算符中的任意一个时,通常也需显式定义其他四个。1. 析构函数负责释放资源,若未显式定义,编译器会生成合成版本,但管理资源时应自定义并正确释放;2. 拷贝构造函数与拷贝赋值运算符需实现深拷贝以避免资源共享问题,可使用复制并交换技术简化逻辑;3. 移动构造函数与移动赋值运算符用于提升性能,应标记为noexcept并将原对象置为空状态;4. 若不希望类被拷贝或移动,应使用= delete明确禁用相应操作,以在编译期阻止错误使用;5. 根据实际需求决定是否自定义或删除这些函数,确保类的安全性与可控性。

怎样编写C++中的拷贝控制成员 三五法则实践指南

在C++中,类的拷贝控制成员函数是管理对象生命周期和资源的关键部分。如果你需要自定义其中一个(比如析构函数、拷贝构造函数或拷贝赋值运算符),那么很可能你也需要自定义其他几个。这就是所谓的“三五法则”——当你定义了五个中的任何一个(包括移动构造函数和移动赋值运算符),最好明确地定义所有五个。

怎样编写C++中的拷贝控制成员 三五法则实践指南

下面是一些实用建议,帮助你正确编写这些成员函数。

怎样编写C++中的拷贝控制成员 三五法则实践指南

析构函数:释放资源的基本保障

如果你的类管理了某种资源(比如动态内存、文件句柄等),你需要自定义析构函数来释放这些资源。

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

class MyClass {
    int* data;
public:
    MyClass() : data(new int[100]) {}
    ~MyClass() {
        delete[] data;  // 正确释放资源
    }
};

注意点:

怎样编写C++中的拷贝控制成员 三五法则实践指南
  • 如果你没有显式定义析构函数,编译器会生成一个合成版本。
  • 如果你定义了析构函数,通常也意味着你需要自己处理拷贝和移动操作。
  • 如果你的类不需要释放资源,可以使用 = default 来让编译器生成默认版本。

拷贝构造函数与拷贝赋值运算符:实现深拷贝

如果你的类包含指针或其他需要手动管理的资源,必须实现深拷贝逻辑,否则默认的按成员拷贝会导致多个对象共享同一块资源,最终引发重复释放等问题。

STORYD
STORYD

帮你写出让领导满意的精美文稿

下载
class MyClass {
    int* data;
public:
    MyClass(const MyClass& other) {
        data = new int[100];
        std::copy(other.data, other.data + 100, data);
    }

    MyClass& operator=(const MyClass& other) {
        if (this == &other) return *this;

        int* newData = new int[100];
        std::copy(other.data, other.data + 100, newData);
        delete[] data;
        data = newData;
        return *this;
    }
};

常见做法:

  • 使用“复制并交换”技术简化赋值操作。
  • 避免浅拷贝导致的悬空指针问题。
  • 考虑是否允许拷贝,如果不允许,就用 = delete 明确禁用。

移动构造函数与移动赋值运算符:提升性能

C++11引入了移动语义,用于避免不必要的拷贝。如果你的类管理资源,并且希望支持高效的移动操作,就需要实现这两个函数。

class MyClass {
    int* data;
public:
    MyClass(MyClass&& other) noexcept {
        data = other.data;
        other.data = nullptr;
    }

    MyClass& operator=(MyClass&& other) noexcept {
        if (this == &other) return *this;

        delete[] data;
        data = other.data;
        other.data = nullptr;
        return *this;
    }
};

注意事项:

  • 移动操作通常应标记为 noexcept,以支持某些标准库容器的操作优化。
  • 如果你不打算支持移动,可以删除它们或者依赖编译器生成的版本。
  • 实现移动时要记得将原对象置为空状态,防止后续释放野指针。

是否应该删除某些操作?

有时候你并不希望某个类被拷贝或移动,比如单例类或资源唯一持有者。这时候可以用 = delete 来显式禁止:

class NoCopy {
public:
    NoCopy() = default;
    NoCopy(const NoCopy&) = delete;
    NoCopy& operator=(const NoCopy&) = delete;
};

这样做有两个好处:

  • 明确表达设计意图;
  • 编译时报错,而不是运行时出问题。

基本上就这些。掌握这五类函数的编写原则,能让你写出更安全、更可控的C++类。关键是理解每个函数的作用,并根据实际需求决定是否自定义、是否删除。

相关专题

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

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

1435

2023.10.24

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

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

223

2024.02.23

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

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

84

2025.10.17

空指针异常处理
空指针异常处理

本专题整合了空指针异常解决方法,阅读专题下面的文章了解更多详细内容。

20

2025.11.16

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

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

266

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

207

2023.12.29

default gateway怎么配置
default gateway怎么配置

配置default gateway的步骤:1、了解网络环境;2、获取路由器IP地址;3、登录路由器管理界面;4、找到并配置WAN口设置;5、配置默认网关;6、保存设置并退出;7、检查网络连接是否正常。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

219

2023.12.07

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

3

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

1

2025.12.31

热门下载

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

精品课程

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

共94课时 | 5.7万人学习

C 教程
C 教程

共75课时 | 3.8万人学习

C++教程
C++教程

共115课时 | 10.6万人学习

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

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