首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >是否存在用push_back替换emplace_back不正确的情况?

是否存在用push_back替换emplace_back不正确的情况?
EN

Stack Overflow用户
提问于 2014-02-27 21:02:05
回答 5查看 1.5K关注 0票数 9

我可以通过用C++03替换std::vector::push_back并用C++ 11编译器编译来破坏一个有效的emplace_back程序吗?通过阅读emplace_back引用,我认为这是不应该发生的,但我承认我没有完全获得rvalue引用。

EN

回答 5

Stack Overflow用户

回答已采纳

发布于 2014-02-27 23:45:37

我构造了一个简短的示例,当push_backemplace_back替换时,它实际上无法编译。

代码语言:javascript
运行
复制
#include <vector>
struct S {
    S(double) {}
  private:
    explicit S(int) {}
};
int main() {
    std::vector<S>().push_back(0); // OK
    std::vector<S>().emplace_back(0); // error!
}

push_back的调用需要将其参数0int类型转换为S类型。因为这是一个隐式转换,所以不考虑显式构造函数S::S(int),并且调用S::S(double)。另一方面,emplace_back执行直接初始化,因此考虑了S::S(double)S::S(int)。后者是一个更好的匹配,但它是private,所以程序是错误的.

票数 16
EN

Stack Overflow用户

发布于 2014-07-20 02:59:18

在异常情况下,emplace版本根本不创建所需类型的对象。这可能会导致一个错误。

考虑下面的示例,它使用std::vector来简化操作(假设uptr的行为类似于std::unique_ptr,除非构造函数不是显式的):

代码语言:javascript
运行
复制
std::vector<uptr<T>> vec;
vec.push_back(new T());

它是例外安全的。创建一个临时uptr<T>以传递给push_back,后者被移动到向量中。如果向量重新分配失败,则分配的T仍然由一个正确删除它的智能指针所拥有。

与之相比:

代码语言:javascript
运行
复制
std::vector<uptr<T>> vec;
vec.emplace_back(new T());

不允许emplace_back创建临时对象。ptr将在向量中创建一次.如果重新分配失败,就没有就地创建的位置,也不会创建任何智能指针。T将被泄露。

当然,最好的选择是:

代码语言:javascript
运行
复制
std::vector<std::unique_ptr<T>> vec;
vec.push_back(make_unique<T>());

这相当于第一个,但使智能指针创建显式。

票数 2
EN

Stack Overflow用户

发布于 2014-02-27 21:08:09

如果您在向量中持有的对象的复制构造函数中没有疯狂的副作用,则不会。

引入emplace_back是为了优化不必要的复制和移动。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/22080290

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档