零/三/五法则是C++中关于资源管理的指导原则:若需自定义析构函数、拷贝构造、拷贝赋值、移动构造或移动赋值中的任一函数,通常需显式定义全部。其演进分为三阶段:零法则指不管理资源时可依赖编译器默认生成;三法则要求自定义析构函数时也实现拷贝构造与拷贝赋值;五法则在C++11后补充移动构造与移动赋值。该法则防止浅拷贝导致的重复释放问题,确保资源安全释放、深拷贝与高效移动。例如手动管理内存的MyString类需实现全部五个函数,而现代C++推荐使用std::string等RAII类型,使类无需自定义任何特殊成员函数,遵循零法则,提升安全性与简洁性。

在C++中,“零/三/五法则”是关于类的特殊成员函数管理资源时的一套经验性规则,用来指导程序员正确实现类的拷贝控制和资源管理。这个法则并不是语言标准中的硬性规定,而是一种编程实践中的最佳建议。
什么是零/三/五法则?
“零/三/五法则”指的是:如果你需要手动定义以下五个特殊成员函数中的任何一个,那么你很可能需要明确地定义全部或大部分:
- 析构函数(destructor)
- 拷贝构造函数(copy constructor)
- 拷贝赋值运算符(copy assignment operator)
- 移动构造函数(move constructor)
- 移动赋值运算符(move assignment operator)
该法则分为三个阶段演进:
• 零法则:如果类不需要自定义析构函数,意味着它不直接管理资源(如裸指针、文件句柄等),可以完全依赖编译器生成的默认拷贝和移动操作。• 三法则:如果类需要自定义析构函数,通常也需要自定义拷贝构造函数和拷贝赋值运算符(C++11之前)。
• 五法则:在C++11引入移动语义后,若类管理资源并需要自定义析构函数,则还应考虑实现移动构造函数和移动赋值运算符。
为什么需要这组法则?
当类管理动态资源(比如用 raw pointer 指向堆内存),使用默认的拷贝行为会导致浅拷贝问题:
立即学习“C++免费学习笔记(深入)”;
- 两个对象指向同一块内存,析构时可能重复释放,引发未定义行为。
- 默认生成的操作无法正确处理资源的所有权转移。
通过手动定义这些函数,可以确保:
- 资源被正确释放(析构函数)
- 拷贝时进行深拷贝(拷贝构造与赋值)
- 移动时高效转移资源所有权(移动操作)
实际应用示例
假设我们写一个简单的字符串类,使用原始指针管理字符数组:
class MyString {
char* data;
size_t size;
public:
// 构造函数
MyString(const char* str = "") {
size = std::strlen(str);
data = new char[size + 1];
std::strcpy(data, str);
}
// 1. 自定义析构函数
~MyString() { delete[] data; }
// 2. 拷贝构造函数(深拷贝)
MyString(const MyString& other) : size(other.size) {
data = new char[size + 1];
std::strcpy(data, other.data);
}
// 3. 拷贝赋值运算符
MyString& operator=(const MyString& other) {
if (this != &other) {
delete[] data;
size = other.size;
data = new char[size + 1];
std::strcpy(data, other.data);
}
return *this;
}
// 4. 移动构造函数
MyString(MyString&& other) noexcept
: data(other.data), size(other.size) {
other.data = nullptr;
other.size = 0;
}
// 5. 移动赋值运算符
MyString& operator=(MyString&& other) noexcept {
if (this != &other) {
delete[] data;
data = other.data;
size = other.size;
other.data = nullptr;
other.size = 0;
}
return *this;
}
};
这个类遵循了“五法则”,能安全地管理资源,并支持高效的移动操作。
现代C++中的简化方式
在现代C++中,更推荐使用RAII容器(如 std::string、std::unique_ptr、std::vector)来管理资源。这样就可以避免手动编写这些函数,从而遵守“零法则”:
class MyStringModern {
std::string data; // 使用标准库管理资源
public:
MyStringModern(const std::string& str) : data(str) {}
// 不需要自定义析构、拷贝、移动函数!
// 编译器生成的默认版本已经足够安全高效。
};
只要成员本身正确实现了资源管理,外层类就可以安全地依赖默认行为。
基本上就这些。掌握零/三/五法则是写出安全、高效C++类的关键基础。优先使用标准库类型,减少手动资源管理,能大幅降低出错概率。










