0

0

C++的三/五/零法则是什么_C++类中拷贝/移动构造和赋值运算符的编写规则

冰火之心

冰火之心

发布时间:2026-01-04 19:31:52

|

551人浏览过

|

来源于php中文网

原创

三/五/零法则是C++中关于特殊成员函数的设计原则:若需自定义析构函数、拷贝或移动操作中的任一个,就应显式定义全部五个(析构、拷贝构造、拷贝赋值、移动构造、移动赋值),或完全不定义而依赖编译器生成,推荐使用RAII如vector、智能指针实现“零”手动管理,避免资源泄漏。

c++的三/五/零法则是什么_c++类中拷贝/移动构造和赋值运算符的编写规则

在C++中,三/五/零法则是关于类中特殊成员函数设计的重要指导原则,它帮助开发者正确管理资源,避免内存泄漏或未定义行为。这个法则的核心是:如果你需要手动定义以下五个特殊成员函数中的任意一个,那么你很可能需要显式定义全部五个,或者干脆一个都不定义(即遵循“零”规则)。

三/五/零法则详解

这五个特殊成员函数包括:

  • 析构函数(destructor)
  • 拷贝构造函数(copy constructor)
  • 拷贝赋值运算符(copy assignment operator)
  • 移动构造函数(move constructor)
  • 移动赋值运算符(move assignment operator)

该法则的含义如下:

三法则:在C++11之前,只有前三个函数重要。如果你需要自定义其中一个(比如因为类中管理了动态内存),通常也需要自定义另外两个,以确保资源被正确复制和释放。

五法则:C++11引入了移动语义后,如果类需要自定义析构函数、拷贝操作或移动操作中的任何一个,就应当显式定义所有五个函数,否则编译器生成的默认版本可能不符合预期。

零法则:理想情况下,尽量避免手动管理资源。使用RAII(如智能指针、容器等)让编译器自动生成正确的特殊成员函数。此时,你不需要定义任何这些函数——即“零”个。

何时需要手动定义?

当你在类中直接管理原始资源时,例如:

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

  • 使用裸指针指向动态分配的内存(如 new int[10])
  • 持有文件句柄、socket、互斥锁等系统资源
  • 实现自定义的资源生命周期控制逻辑

在这种情况下,编译器提供的默认拷贝行为是浅拷贝,会导致多个对象指向同一块内存,析构时重复释放,引发崩溃。

Baklib
Baklib

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

下载

如何正确实现五函数?

假设你有一个类管理一段动态数组:

class MyArray {
private:
    int* data;
    size_t size;

public: // 构造函数 explicit MyArray(size_t s) : size(s), data(new int[s]{}) {}

// 1. 析构函数
~MyArray() { delete[] data; }

// 2. 拷贝构造函数
MyArray(const MyArray& other)
    : size(other.size), data(new int[other.size])
{
    std::copy(other.data, other.data + size, data);
}

// 3. 拷贝赋值运算符 —— 注意自我赋值安全和异常安全
MyArray& operator=(const MyArray& other) {
    if (this != &other) {
        int* new_data = new int[other.size];
        std::copy(other.data, other.data + other.size, new_data);
        delete[] data;
        data = new_data;
        size = other.size;
    }
    return *this;
}

// 4. 移动构造函数
MyArray(MyArray&& other) noexcept
    : size(other.size), data(other.data)
{
    other.size = 0;
    other.data = nullptr;
}

// 5. 移动赋值运算符
MyArray& operator=(MyArray&& other) noexcept {
    if (this != &other) {
        delete[] data;
        data = other.data;
        size = other.size;
        other.data = nullptr;
        other.size = 0;
    }
    return *this;
}

};

注意点:

  • 移动操作应标记为 noexcept,以便标准库能安全使用它们(如 vector 扩容)
  • 拷贝赋值要处理自赋值问题
  • 移动后原对象应处于“有效但可析构”状态

推荐做法:优先使用零法则

大多数情况下,你应该用更高级的抽象代替手动资源管理:

class MyArrayBetter {
private:
    std::vector data;  // 使用 vector 自动管理内存

public: explicit MyArrayBetter(size_t s) : data(s, 0) {} // 不需要手动定义析构、拷贝、移动函数! };

此时,编译器会自动生成正确且高效的拷贝和移动操作,你无需操心细节。

基本上就这些。掌握三/五/零法则,能让你写出更安全、更现代的C++代码。重点不是死记硬背五个函数怎么写,而是理解资源管理的本质:要么全权掌控,要么完全交出。不复杂但容易忽略。

相关专题

更多
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

string转int
string转int

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

314

2023.08.02

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

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

527

2024.08.29

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

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

49

2025.08.29

C++中int的含义
C++中int的含义

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

193

2025.08.29

漫蛙2入口地址合集
漫蛙2入口地址合集

本专题整合了漫蛙2入口汇总,阅读专题下面的文章了解更多详细内容。

148

2026.01.06

AO3中文版地址汇总
AO3中文版地址汇总

本专题整合了AO3中文版地址合集,阅读专题下面的文章了解更多详细内容。

82

2026.01.06

热门下载

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

精品课程

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

共94课时 | 6.2万人学习

C 教程
C 教程

共75课时 | 3.9万人学习

C++教程
C++教程

共115课时 | 11.3万人学习

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

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