resolve conflict

This commit is contained in:
johnwdjiang 2020-10-13 15:32:16 +08:00
commit 8eca975896
3 changed files with 299 additions and 14 deletions

View File

@ -1,26 +1,210 @@
## Item21: 首选std::make_unique和std::make_shared相比直接使用new
## Item 21:优先考虑使用std::make_unique和std::make_shared而非new
让我们首先介绍一下std::make_unique和std::make_sharedstd::make_shared是C++11的部分但是不幸的是std::make_unique是C++14才引入到标准库的。如果你使用的是C++11不用担心因为std::make_unique的基本版本很容易自己实现
让我们先对std::make_unique和std::make_shared做个铺垫。std::make_shared 是C++11标准的一部分但很可惜的是std::make_unique不是。它从C++14开始加入标准库。如果你在使用C++11不用担心一个基础版本的std::make_unique是很容易自己写出的如下
```cpp
template<typename T, typename... Ts>
std::unique_ptr<T> make_unique(Ts&&... params)
{
return std::unique_ptr<T>(new T(std::forward<Ts>(params)...));
return std::unique_ptr<T>(new T(std::forward<Ts>(params)...));
}
```
如你所看到的make_unique只是通过完美转发参数到对象的构造器中被创建从new出来的原始指针构造一个std::unique_ptr的只能指针然后返回。这种形式的函数不支持数组和自定义删除器查看item18但是它表明只需要一点点努力你就可以自行实现需要的make_unique。请记住不要把自己的版本放在命名空间std中因为在升级到C++14之后避免出现冲突。
正如你看到的make_unique只是将它的参数完美转发到所要创建的对象的构造函数从新产生的原始指针里面构造出std::unique_ptr并返回这个std::unique_ptr。这种形式的函数不支持数组和自定义析构但它给出了一个示范只需一点努力就能写出你想要的make_uniqe函数。需要记住的是不要把它放到std命名空间中因为你可能并不希望在升级厂家编译器到符合C++14标准的时候产生冲突。
Std::make_unique和make_shared是三大make函数中的两个接收任意参数集的函数将其完美转发给动态分配对象的构造函数然后返回指向该对象的只能指针。第三个make函数是std::allocate_shared。它的行为就像std::make_shared不同之处在于它的第一个参数是用于动态分配内存的分配器对象。
std::make_unique和std::make_shared有两三个make functions:接收抽象参数完美转发到构造函数去动态分配一个对象然后返回这个指向这个对象的指针。第三个make function 是std::allocate_shared.它和std::make_shared一样除了第一个参数是用来动态分配内存的对象。
使用和不使用make函数进行只能指针创建的最不必要的比较也能揭示为什么此类函数是首选的原因。考虑
即使是对使用和不使用make函数创建智能指针的最简单比较也揭示了为什么最好使用这些函数的第一个原因。例如
```cpp
auto upw1(std::make_unique<Widget>());
std::unique_ptr<Widget> upw2(new Widget);
auto spw1(std::make_shared<Widget>());
std::shared_ptr<Widget> spw2(new Widget);
```c++
auto upw1(std::make_unique<Widget>()); // with make func
std::unique_ptr<Widget> upw2(new Widget); // without make func
auto spw1(std::make_shared<Widget>()); // with make func
std::shared_ptr<Widget> spw2(new Widget); // without make func
```
已经强调了本质区别从写法上使用new版本会重复创建的类型但是make版本不会。重复类型与软件工程的一项主要原则相违背
我高亮了区别使用new的版本重复了类型但是make function的版本没有。(译者注这里高亮的是Widget用new的声明语句需要写2遍Widgetmake function只需要写一次) 重复写类型和软件工程里面一个关键原则相冲突应该避免重复代码。源代码中的重复增加了编译的时间会导致目标代码冗余并且通常会让代码库使用更加困难。它经常演变成不一致的代码而代码库中的不一致常常导致bug。此外打两次字比一次更费力而且谁不喜欢减少打字负担
第二个使用make function的原因和异常安全有段。假设我们有个函数按照某种优先级处理Widget
```c++
void processWidget(std::shared_ptr<Widget> spw, int priority);
```
根据值传递std::shared ptr可能看起来很可疑但是Item 41解释了如果processWidget总是复制std::shared ptr(例如通过将其存储在已处理的Widget的数据结构中),那么这可能是一个可复用的设计选择。
现在假设我们有一个函数来计算相关的优先级
`int computePriority();`
并且我们在调用processWidget时使用了new而不是std:: make_shared
```c++
processWidget(std::shared_ptr<Widget>(new Widget), computePriority()); // potential resource leak!
```
如注释所说这段代码可能在new Widget时发生泄露。为何调用的代码和被调用的函数都用std::shared_ptrs,且std::shared_ptrs就是设计出来防止泄露的。它们会在最后一个std::shared_ptr销毁时自动释放所指向的内存。如果每个人在每个地方都用std::shared_ptrs,这段代码怎么会泄露呢?
答案和编译器将源码转换为目标代码有关。在运行时一个函数的参数必须先被计算才能被调用所以在调用processWidget之前必须执行以下操作processWidget才开始执行
- 表达式'new Widget'必须计算,例如,一个Widget对象必须在堆上被创建
- 负责管理new出来指针的std::shared_ptr<Widget>构造函数必须被执行
- computePriority()必须运行
编译器不需要按照执行顺序生成代码。“new Widget"必须在std::shared_ptr的构造函数被调用前执行因为new出来的结果作为构造函数的参数但compute Priority可能在这之前之后或者之间执行。也就是说编译器可能按照这个执行顺序生成代码
1. 执行new Widget
2. 执行computePriority
3. 运行std::shared_ptr构造函数
如果按照这样生成代码并且在运行是computePriority产生了异常那么第一步动态分配的Widget就会泄露。因为它永远都不会被第三步的std::shared_ptr所管理了。
使用std::make_shared可以防止这种问题。调用代码看起来像是这样
```c++
processWidget(std::make_shared<Widget>(), computePriority());
```
在运行时std::make_shared和computePriority会先被调用。如果是std::make_shared在computePriority调用前动态分配Widget的原始指针会安全的保存在作为返回值的std::shared_ptr中。如果compu tePriority生成一个异常那么std::shared_ptr析构函数将确保管理的Widget被销毁。如果首先调用computePriority并产生一个异常那么std::make_shared将不会被调用因此也就不需要担心new Widget(会泄露)。
如果我们将std::shared_ptr,std::make_shared替换成std::unique_ptr,std::make_unique,同样的道理也适用。因此在编写异常安全代码时使用std::make_unique而不是new与使用std::make_shared同样重要。
std::make_shared的一个特性(与直接使用new相比)得到了效率提升。使用std::make_shared允许编译器生成更小更快的代码并使用更简洁的数据结构。考虑以下对new的直接使用
```c++
std::shared_ptr<Widget> spw(new Widget);
```
显然,这段代码需要进行内存分配,但它实际上执行了两次.Item 19解释了每个std::shared_ptr指向一个控制块其中包含被指向对象的引用计数。这个控制块的内存在std::shared_ptr构造函数中分配。因此直接使用new需要为Widget分配一次内存为控制块分配再分配一次内存。
如果使用std::make_shared代替` auto spw = std::make_shared_ptr<Widget>();`一次分配足矣。这是因为std::make_shared分配一块内存同时容纳了Widget对象和控制块。这种优化减少了程序的静态大小因为代码只包含一个内存分配调用并且它提高了可执行代码的速度因为内存只分配一次。此外使用std::make_shared避免了对控制块中的某些簿记信息的需要潜在地减少了程序的总内存占用。
对于std::make_shared的效率分析同样适用于std::allocate_shared因此std::make_shared的性能优势也扩展到了该函数。
更倾向于使用函数而不是直接使用new的争论非常激烈。尽管它们在软件工程、异常安全和效率方面具有优势但本item的意见是更倾向于使用make函数而不是完全依赖于它们。这是因为有些情况下它们不能或不应该被使用。
例如没有make函数允许指定定制的析构(见item18和19),但是std::unique_ptr和std::shared_ptr有构造函数这么做。给Widget自定义一个析构:
```
auto widgetDeleter = [](Widget*){...};
```
使用new创建智能指针非常简单:
```
std::unique_ptr<Widget, decltype(widgetDeleter)>
upw(new Widget, widgetDeleter);
std::shared_ptr<Widget> spw(new Widget, widgetDeleter);
```
对于make函数没有办法做同样的事情。
make函数第二个限制来自于其单一概念的句法细节。Item7解释了当构造函数重载有std::initializer_list作为参数和不用其作为参数时用大括号创建对象更倾向于使用std::initializer_list作为参数的构造函数而用圆括号创建对象倾向于不用std::initializer_list作为参数的构造函数。make函数会将它们的参数完美转发给对象构造函数但是它们是使用圆括号还是大括号对某些类型问题的答案会很不相同。例如在这些调用中
```
auto upv = std::make_unique<std::vector<int>>(10, 20);
auto spv = std::make_shared<std::vector<int>>(10, 20);
```
生成的智能指针是否指向带有10个元素的std::vector每个元素值为20或指向带有两个元素的std::vector其中一个元素值10另一个为20 ?或者结果是不确定的?
好消息是这并非不确定两种调用都创建了10个元素每个值为20.这意味着在make函数中完美转发使用圆括号而不是大括号。坏消息是如果你想用大括号初始化指向的对象你必须直接使用new。使用make函数需要能够完美转发大括号初始化但是正如item31所说大括号初始化无法完美转发。但是item30介绍了一个变通的方法使用auto类型推导从大括号初始化创建std::initializer_list对象(见Item 2)然后将auto创建的对象传递给make函数。
```
// create std::initializer_list
auto initList = { 10, 20 };
// create std::vector using std::initializer_list ctor
auto spv = std::make_shared<std::vector<int>>(initList);
```
对于std::unique_ptr,只有这两种情景定制删除和大括号初始化使用make函数有点问题。对于std::shared_ptr和它的make函数还有至少2个问题。都属于边界问题但是一些开发者常碰到你也可能是其中之一。
一些类重载了operator new和operator delete。这些函数的存在意味着对这些类型的对象的全局内存分配和释放是不合常规的。设计这种定制类往往只会精确的分配、释放对象的大小。例如Widget类的operator new和operator delete只会处理sizeof(Widget)大小的内存块的分配和释放。这种常识不太适用于std::shared_ptr对定制化分配(通过std::allocate_shared)和释放(通过定制化deleters)因为std::allocate_shared需要的内存总大小不等于动态分配的对象大小还需要再加上控制块大小。因此适用make函数去创建重载了operator new 和 operator delete类的对象是个典型的糟糕想法。
与直接使用new相比std::make_shared在大小和速度上的优势源于std::shared_ptr的控制块与指向的对象放在同一块内存中。当对象的引用计数降为0对象被销毁(析构函数被调用).但是,因为控制块和对象被放在同一块分配的内存块中,直到控制块的内存也被销毁,它占用的内存是不会被释放的。
正如我说控制块除了引用计数还包含簿记信息。引用计数追踪有多少std::shared_ptrs指向控制块但控制块还有第二个计数记录多少个std::weak_ptrs指向控制块。第二个引用计数就是weak count。当一个std::weak_ptr检测对象是否过期时(见item 19),它会检测指向的控制块中的引用计数(而不是weak count)。如果引用计数是0(即对象没有std::shared_ptr再指向它已经被销毁了)std::weak_ptr已经过期。否则就没过期。
只要std::weak_ptrs引用一个控制块(即weak count大于零)该控制块必须继续存在。只要控制块存在包含它的内存就必须保持分配。通过std::shared_ptr make函数分配的内存直到最后一个std::shared_ptr和最后一个指向它的std::weak_ptr已被销毁才会释放。
如果对象类型非常大而且销毁最后一个std::shared_ptr和销毁最后一个std::weak_ptr之间的时间很长那么在销毁对象和释放它所占用的内存之间可能会出现延迟。
```c++
class ReallyBigType { … };
// 通过std::make_shared创建一个大对象
auto pBigObj = std::make_shared<ReallyBigType>();
… // 创建 std::shared_ptrs 和 std::weak_ptrs
// 指向这个对象,使用它们
… // 最后一个 std::shared_ptr 在这销毁,
// 但 std::weak_ptrs 还在
… // 在这个阶段,原来分配给大对象的内存还分配着
… // 最后一个std::weak_ptr在这里销毁;
// 控制块和对象的内存被释放
```
直接只用new一旦最后一个std::shared_ptr被销毁ReallyBigType对象的内存就会被释放
```c++
class ReallyBigType { … };
//通过new创建特大对象
std::shared_ptr<ReallyBigType> pBigObj(new ReallyBigType);
… // 像之前一样,创建 std::shared_ptrs 和 std::weak_ptrs
// 指向这个对象,使用它们
… // 最后一个 std::shared_ptr 在这销毁,
// 但 std::weak_ptrs 还在
// memory for object is deallocated
… // 在这阶段,只有控制块的内存仍然保持分配
… // 最后一个std::weak_ptr在这里销毁;
// 控制块内存被释放
```
如果你发现自己处于不可能或不合适使用std::make_shared的情况下你将想要保证自己不受我们之前看到的异常安全问题的影响。最好的方法是确保在直接使用new时在一个不做其他事情的语句中立即将结果传递到智能指针构造函数。这可以防止编译器生成的代码在使用new和调用管理新对象的智能指针的构造函数之间发生异常。
例如考虑我们前面讨论过的processWidget函数对其非异常安全调用的一个小修改。这一次我们将指定一个自定义删除器:
```c++
void processWidget(std::shared_ptr<Widget> spw, int priority);
void cusDel(Widget *ptr); // 自定义删除器
```
这是非异常安全调用:
```c++
//和之前一样,潜在的内存泄露
processWidget(
std::shared_ptr<Widget>(new Widget, cusDel),
computePriority()
);
```
回想一下:如果computePriority在“new Widget”之后而在std::shared_ptr构造函数之前调用并且如果computePriority产生一个异常那么动态分配的Widget将会泄漏。
这里使用自定义删除排除了对std::make_shared的使用因此避免这个问题的方法是将Widget的分配和std::shared_ptr的构造放入它们自己的语句中然后使用得到的std::shared_ptr调用processWidget。这是该技术的本质不过正如我们稍后将看到的我们可以对其进行调整以提高其性能
```c++
std::shared_ptr<Widget> spw(new Widget, cusDel);
processWidget(spw, computePriority()); // 正确,但是没优化,见下
```
这是可行的因为std::shared_ptr假定了传递给它的构造函数的原始指针的所有权即使构造函数产生了一个异常。此例中如果spw的构造函数抛出异常(即无法为控制块动态分配内存)仍然能够保证cusDel会在new Widget产生的指针上调用。
一个小小的性能问题是在异常不安全调用中我们将一个右值传递给processWidget
```c++
processWidget(
std::shared_ptr<Widget>(new Widget, cusDel), // arg is rvalue
computePriority()
);
```
但是在异常安全调用中,我们传递了左值
```c++
processWidget(spw, computePriority()); //spw是左值
```
因为processWidget的std::shared_ptr参数是传值传右值给构造函数只需要move而传递左值需要拷贝。对std::shared_ptr而言这种区别是有意义的因为拷贝std::shared_ptr需要对引用计数原子加move则不需要对引用计数有操作。为了使异常安全代码达到异常不安全代码的性能水平我们需要用std::move将spw转换为右值.
```c++
processWidget(std::move(spw), computePriority());
```
这很有趣也值得了解但通常是无关紧要的因为您很少有理由不使用make函数。除非你有令人信服的理由这样做否则你应该使用make函数。
记住:
- 和直接使用new相比make函数消除了代码重复提高了异常安全性。对于std::make_shared和std::allocate_shared,生成的代码更小更快。
- 不适合使用make函数的情况包括需要指定自定义删除器和希望用大括号初始化
- 对于std::shared_ptrs, make函数可能不被建议的其他情况包括(1)有自定义内存管理的类和(2)特别关注内存的系统非常大的对象以及std::weak_ptrs比对应的std::shared_ptrs活得更久

View File

@ -0,0 +1,101 @@
# 对于std::forward的auto&&形参使用decltype
泛型lambda(generic lambdas)是C++14中最值得期待的特性之一——因为在lambda的参数中可以使用auto关键字。这个特性的实现是非常直截了当的即在闭包类中的operator()函数是一个函数模版。例如存在这么一个lambda
```c++
auto f = [](auto x){ return func(normalize(x)); };
```
对应的闭包类中的函数调用操作符看来就变成这样:
```c++
class SomeCompilerGeneratedClassName { public:
template<typename T>
auto operator()(T x) const
{ return func(normalize(x)); }
...
};
```
在这个样例中lambda对变量`x`做的唯一一件事就是把它转发给函数`normalize`。如果函数`normalize`对待左值右值的方式不一样这个lambda的实现方式就不大合适了因为即使传递到lambda的实参是一个右值lambda传递进去的形参总是一个左值。
实现这个lambda的正确方式是把`x`完美转发给函数`normalize`。这样做需要对代码做两处修改。首先x需要改成通用引用其次需要使用`std::forward`将`x`转发到函数`normalize`。实际上的修改如下:
```c++
auto f = [](auto&& x)
{ return func(normalize(std::forward<???>(x))); };
```
在理论和实际之间存在一个问题:你传递给`std::forward`的参数是什么类型,就决定了上面的`???`该怎么修改。
一般来说,当你在使用完美转发时,你是在一个接受类型参数为`T`的模版函数里,所以你可以写`std::forward<T>`。但在泛型lambda中没有可用的类型参数`T`。在lambda生成的闭包里模版化的`operator()`函数中的确有一个`T`但在lambda里却无法直接使用它。
前面item28解释过在传递给通用引用的是一个左值那么它会变成左值引用。传递的是右值就会变成右值引用。这意味着在这个lambda中可以通过检查`x`的类型来检查传递进来的实参是一个左值还是右值decltype就可以实现这样的效果。传递给lambda的是一个左值`decltype(x)`就能产生一个左值引用;如果传递的是一个右值,`decltype(x)`就会产生右值引用。
Item28也解释过在调用`std::forward`传递给它的类型类型参数是一个左值引用时会返回一个左值传递的是一个非引用类型时返回的是一个右值引用而不是常规的非引用。在前面的lambda中如果x绑定的是一个左值引用`decltype(x)`就能产生一个左值引用;如果绑定的是一个右值,`decltype(x)`就会产生右值引用,而不是常规的非引用。
在看一下Item28中关于`std::forward`的C++14实现
```c++
template<typename T> // in namespace
T&& forward(remove_reference_t<T>& param) // std
{
return static_cast<T&&>(param);
}
```
如果用户想要完美转发一个Widget类型的右值时它会使用Widget类型非引用类型来示例化`std::forward`,然后产生以下的函数:
```c++
Widget&& forward(Widget& param)
{ // instantiation of
return static_cast<Widget&&>(param); // std::forward when
} // T is Widget
```
思考一下如果用户代码想要完美转发一个Widget类型的右值但没有遵守规则将T指定为非引用类型而是将T指定为右值引用这回发生什么思考将T换成`Widget`如何,在`std::forward`实例化、应用了`remove_reference_t`后,音乐折叠之前,这是产生的代码:
```c++
Widget&& && forward(Widget& param) // instantiation of
{ // std::forward when
return static_cast<Widget&& &&>(param); // T is Widget&&
} // (before reference-collapsing)
```
应用了引用折叠之后,代码会变成:
```c++
Widget&& forward(Widget& param) // instantiation of
{ // std::forward when
return static_cast<Widget&&>(param); // T is Widget&&
} // (before reference-collapsing)
```
对比发现,用一个右值引用去实例化`std::forward`和用非引用类型去实例化产生的结果是一样的。
那是一个很好的消息引用当传递给lambda形参x的是一个右值实参时`decltype(x)`可以产生一个右值引用。前面已经确认过把一个左值传给lambda时`decltype(x)`会产生一个可以传给`std::forward`的常规类型。而现在也验证了对于右值,把`decltype(x)`产生的类型传递给`std::forward`的类型参数是非传统的,不过它产生的实例化结果与传统类型相同。所以无论是左值还是右值,把`decltype(x)`传递给`std::forward`都能得到我们想要的结果因此lambda的完美转发可以写成
```c++
auto f =
[](auto&& param)
{
return
func(normalize(std::forward<decltype(pram)>(param)));
};
```
再加上6个点就可以让我们的lambda完美转发接受多个参数了因为C++14中的lambda参数是可变的
```c++
auto f =
[](auto&&... params)
{
return
func(normalized(std::forward<decltype(params)>(params)...));
};
```
要谨记的是:
* 对`auto&&`参数使用`decltype`来(`std::forward`)转发参数;

View File

@ -37,7 +37,7 @@
1. [Item 18:对于独占资源使用std::unique_ptr](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/4.SmartPointers/item18.md) __@wendajiang更新完成__
2. [Item 19:对于共享资源使用std::shared_ptr](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/4.SmartPointers/item19.md) __已修订__
3. [Item 20:像std::shared_ptr一样使用std::weak_ptr可能造成dangle](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/4.SmartPointers/item20.md) __更新中__
4. Item 21:优先考虑使用std::make_unique和std::make_shared而非new
4. [Item 21:优先考虑使用std::make_unique和std::make_shared而非new](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/4.SmartPointers/item21.md) __由 @pusidun贡献__
5. [Item 22:当使用Pimpl惯用法请在实现文件中定义特殊成员函数](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/4.SmartPointers/item22.md) __由 @BlurryLight贡献__
5. 右值引用,移动语意,完美转发
1. [Item 23:理解std::move和std::forward](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/5.RvalueReferences_MovingSemantics_And_PerfectForwarding/item23.md) __由 @BlurryLight贡献__
@ -51,7 +51,7 @@
6. Lambda表达式
1. [Item 31:避免使用默认捕获模式](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/6.Lambda%20Expressions/item31.md) __由 @LucienXian贡献__
2. [Item 32:使用初始化捕获来移动对象到闭包中](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/6.Lambda%20Expressions/item32.md) __由 @LucienXian贡献__
3. Item 33:对于std::forward的auto&&形参使用decltype
3. [Item 33:对于std::forward的auto&&形参使用decltype](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/6.Lambda%20Expressions/item33.md) __由 @LucienXian贡献__
4. Item 34:有限考虑lambda表达式而非std::bind
7. 并发API
1. [Item 35:优先考虑基于任务的编程而非基于线程的编程](https://github.com/kelthuzadx/EffectiveModernCppChinese/blob/master/7.The%20Concurrency%20API/item35.md) __由 @wendajiang贡献__