item17:complete

This commit is contained in:
kelthuzadx 2019-06-28 19:39:31 +08:00
parent f2d681b67d
commit c5ceccc449

View File

@ -1,4 +1,4 @@
## Item 17:理解特殊成员函数函数的生成
## Ite 17:理解特殊成员函数函数的生成
条款 17:理解特殊成员函数函数的生成
在C++术语中特殊成员函数是指C++自己生成的函数。C++98有四个默认构造函数函数析构函数拷贝构造函数拷贝赋值运算符。这些函数仅在需要的时候才生成比如某个代码使用它们但是它们没有在类中声明。默认构造函数仅在类完全没有构造函数的时候才生成。防止编译器为某个类生成构造函数但是你希望那个构造函数有参数生成的特殊成员函数是隐式public且inline除非该类是继承自某个具有虚函数的类否则生成的析构函数是非虚的。
@ -28,3 +28,91 @@ public:
也许你早已听过_Rule of Three_规则。这个规则告诉我们如果你声明了拷贝构造函数拷贝赋值运算符或者析构函数三者之一你应该也声明其余两个。它来源于长期的观察即用户接管拷贝操作的需求几乎都是因为该类会做其他资源的管理这也几乎意味着1无论哪种资源管理如果能在一个拷贝操作内完成也应该在另一个拷贝操作内完成2类析构函数也需要参与资源的管理通常是释放。通常意义的资源管理指的是内存如STL容器会动态管理内存这也是为什么标准库里面那些管理内存的类都声明了“the big three”拷贝构造拷贝赋值和析构。
**Rule of Three**带来的后果就是只要出现用户定义的析构函数就意味着简单的逐成员拷贝操作不适用于该类。接着如果一个类声明了析构也意味着拷贝操作可能不应该自定生成因为它们做的事情可能是错误的。在C++98提出的时候上述推理没有得倒足够的重视所以C++98用户声明析构不会左右编译器生成拷贝操作的意愿。C++11中情况仍然如此但仅仅是因为限制拷贝操作生成的条件会破坏老代码。
**Rule of Three**规则背后的解释依然有效再加上对声明拷贝操作阻止移动操作隐式生成的观察使得C++11不会为那些有用户定义的析构函数的类生成移动操作。所以仅当下面条件成立时才会生成移动操作
+ 类中没有拷贝操作
+ 类中没有移动操作
+ 类中没有用户定义的析构
有时类似的规则也会扩展至移动操作上面因为现在类声明了拷贝操作C++11不会为它们自动生成其他拷贝操作。这意味着如果你的某个声明了析构或者拷贝的类依赖自动生成的拷贝操作你应该考虑升级这些类消除依赖。假设编译器生成的函数行为是正确的即逐成员拷贝类数据是你期望的行为你的工作很简单C++11的`=default`就可以表达你想做的:
```cpp
class Widget {
public:
...
~Widget();
...
Widget(const Widget&) = default;
Widget&
operator=(const Widget&) = default; // behavior is OK
...
};
```
这种方法通常在多态基类中很有用即根据继承自哪个类来定义接口。多态基类通常有一个虚析构函数因为如果它们非虚一些操作比如对一个基类指针或者引用使用delete或者typeid会产生未定义或错误结果。除非类继承自一个已经是virtual的析构函数否则要想析构为虚函数的唯一方法就是加上virtual关键字。通常默认实现是对的`=default`是一个不错的方式表达默认实现。然而用户声明的析构函数会抑制编译器生成移动操作,所以如果该类需要具有移动性,就为移动操作加上`=default`。声明移动会抑制拷贝生成,所以如果拷贝性也需要支持,再为拷贝操作加上`=default`
```cpp
class Base {
public:
virtual ~Base() = default;
Base(Base&&) = default;
Base& operator=(Base&&) = default;
Base(const Base&) = default;
Base& operator=(const Base&) = default;
...
};
```
实际上,就算编译器乐于为你的类生成拷贝和移动操作,生成的函数也如你所愿,你也应该手动声明它们然后加上`=default`。这看起来比较多余但是它让你的意图更明确也能帮助你避免一些微妙的bug。比如你有一个字符串哈希表即键为整数id值为字符串支持快速查找的数据结构
```cpp
class StringTable {
public:
StringTable() {}
...
private:
std::map<int, std::string> values;
};
```
假设这个类没有声明拷贝操作,没有移动操作,也没有析构,如果它们被用到编译器会自动生成。没错,很方便。
后来需要在对象构造和析构中打日志,增加这种功能很简单:
```cpp
class StringTable {
public:
StringTable()
{ makeLogEntry("Creating StringTable object"); }
~StringTable()
{ makeLogEntry("Destroying StringTable object"); }
...
Item 17 | 113
private:
std::map<int, std::string> values; // as before
};
```
看起来合情合理,但是声明析构有潜在的副作用:它阻止了移动操作的生成。然而,拷贝操作的生成是不受影响的。因此代码能通过编译,运行,也能通过功能(译注:即打日志的功能)测试。功能测试也包括移动功能,因为即使该类不支持移动操作,对该类的移动请求也能通过编译和运行。这个请求正如之前提到的,会转而由拷贝操作完成。它因为着对**StringTable**对象的移动实际上是对对象的拷贝,即拷贝里面的`std::map<int, std::string>`对象。拷贝`std::map<int, std::string>`对象很可能比移动慢几个数量级。简单的加个析构就引入了极大的性能问题!对拷贝和移动操作显式加个`=default`,问题将不再出现。
受够了我喋喋不休的讲述C++11拷贝移动规则了吧你可能想知道什么时候我才会把注意力转入到剩下两个特殊成员函数默认构造和析构。现在就是时候了但是只有一句话因为它们几乎没有改变它们在C++98中是什么样在C++11中就是什么样。
C++11对于特殊成员函数处理的规则如下
+ 默认构造函数和C++98规则相同。仅当类不存在用户声明的构造函数时才自动生成
+ 析构函数基本上和C++98相同稍微不同的是现在析构默认**noexcept**参见Item14。和C++98一样仅当基类析构为虚函数时该类析构才为虚函数。
+ 拷贝构造函数和C++98运行时行为一样逐成员拷贝非static数据。仅当类没有用户定义的拷贝构造时才生成。如果类声明了移动操作它就是**delete**。当用户声明了拷贝赋值或者析构,该函数不再自动生成。
+ 拷贝赋值运算符和C++98运行时行为一样逐成员拷贝赋值非static数据。仅当类没有用户定义的拷贝赋值时才生成。如果类声明了移动操作它就是**delete**。当用户声明了拷贝构造或者析构,该函数不再自动生成。
+ 移动构造函数和移动赋值运算符都对非static数据执行逐成员移动。仅当类没有用户定义的拷贝操作移动操作或析构时才自动生成。
注意没有成员函数模版阻止编译器生成特殊成员函数的规则。这意味着如果**Widget**是这样:
```cpp
class Widget {
...
template<typename T>
Widget(const T& rhs);
template<typename T>
Widget& operator=(const T& rhs); ...
};
```
编译器仍会生成移动和拷贝操作假设正常生成它们的条件满足即使可以模板实例化产出拷贝构造和拷贝赋值运算符的函数签名。当T为Widget时。很可能你会决定这是一个不值得承认的边缘情况但是我提到它是有道理的Item16将会详细讨论它可能带来的后果。
记住:
+ 特殊成员函数是编译器可能自动生成的函数:默认构造,析构,拷贝操作,移动操作
+ 移动操作仅当类没有显式声明移动操作,拷贝操作,析构时才自动生成
+ 拷贝构造仅当类没有显式声明拷贝构造时才自动生成并且如果用户声明了移动操作拷贝构造就是delete。拷贝赋值运算符仅当类没有显式声明拷贝赋值运算符时才自动生成并且如果用户声明了移动操作拷贝赋值运算符就是delete。当用户声明了析构函数拷贝操作不再自动生成