在C++中,当类显式定义某些特殊成员函数时,编译器可能不再自动生成其他相关函数。以下是详细分析:
class MyClass {
public:
// 显式定义拷贝构造函数
MyClass(const MyClass& other) { /* ... */ }
};
int main() {
MyClass obj1; // 错误:默认构造函数被抑制
MyClass obj2(obj1); // 正确:调用拷贝构造函数
return 0;
}显式定义默认构造函数:
class MyClass {
public:
MyClass() = default; // 显式要求编译器生成默认构造函数
MyClass(const MyClass& other) { /* ... */ }
};除了默认构造函数,以下函数的自动生成规则也会受到显式定义其他函数的影响:
显式定义的函数 | 编译器不再自动生成的函数 |
|---|---|
拷贝构造函数 | 默认构造函数、移动构造函数、移动赋值运算符 |
拷贝赋值运算符 | 移动构造函数、移动赋值运算符 |
移动构造函数 | 拷贝构造函数、拷贝赋值运算符 |
移动赋值运算符 | 拷贝构造函数、拷贝赋值运算符 |
析构函数 | 移动构造函数、移动赋值运算符 |
= default 和 = delete为了更精细地控制编译器生成的特殊成员函数,C++11引入了显式语法:
class MyClass {
public:
MyClass() = default; // 强制生成默认构造函数
MyClass(const MyClass&) = default; // 强制生成拷贝构造函数
MyClass(MyClass&&) = default; // 强制生成移动构造函数
};class NonCopyable {
public:
NonCopyable() = default;
NonCopyable(const NonCopyable&) = delete; // 禁止拷贝
NonCopyable& operator=(const NonCopyable&) = delete;
};特殊成员函数 | 自动生成条件 |
|---|---|
默认构造函数 | 没有显式声明任何构造函数 |
拷贝构造函数 | 没有显式声明拷贝构造函数、移动构造函数、移动赋值运算符或析构函数 |
拷贝赋值运算符 | 没有显式声明拷贝赋值运算符、移动构造函数、移动赋值运算符或析构函数 |
移动构造函数 | 没有显式声明拷贝构造函数、拷贝赋值运算符、移动构造函数、移动赋值运算符或析构函数 |
移动赋值运算符 | 没有显式声明拷贝构造函数、拷贝赋值运算符、移动构造函数、移动赋值运算符或析构函数 |
析构函数 | 始终自动生成(除非显式声明) |
= default:保留编译器生成的默认行为,避免手动实现冗余代码。
通过合理控制特殊成员函数的生成,可以避免潜在的错误,并确保类的行为符合预期。