【问题标题】:Exception in std::vector<>::emplace_back() safe?std::vector<>::emplace_back() 中的异常安全吗?
【发布时间】:2017-02-01 09:56:55
【问题描述】:

std::vector&lt;&gt;::emplace_back() 中抛出异常时会发生什么?

例如:

class Foo {
public:
  Foo(int bar) {
    if (bar == 4) throw std::exception("Something went wrong");
  }
}

std::vector<std::unique_ptr<Foo>> foo_list;
foo_list.emplace_back(new Foo(3));
try {
  foo_list.emplace_back(new Foo(4));
} catch (std::exception error) {
  // How bad is it?
}
// Whats inside foo_list now?

我希望向量只包含第一个 Foo 对象。

是这样吗?这是标准保证的吗?

还有:会不会有内存泄漏?

【问题讨论】:

  • emplace_back 在抛出异常时没有运行...
  • 您必须在调用函数之前评估函数参数。构造函数首先抛出。 emplace_back 从未被调用过。
  • @Kenny:有道理,比我想象的要简单!我会在回答时接受它...
  • 标题有点误导,因为正如其他人指出的那样,异常实际上发生在 emplace_back 被调用之前。

标签: c++ c++11 exception vector emplace


【解决方案1】:

我希望向量只包含第一个 Foo 对象。

是这样吗?这是标准保证的吗?

是的。上面的 cmets 已经解释了 emplace_back 甚至不会被调用,因为 Foo 构造函数在初始化函数的参数时抛出。

但是……

还有:会不会有内存泄漏?

是的,您正在使用我在Inserting into a container of smart pointers with emplace_back(new X) 描述的反模式(也发布在Overload Journal #134 - August 2016)。

emplace_back 需要重新分配向量并且由于内存不足而失败时,就会出现问题。传递给函数的指针将丢失,因此您泄漏了Foo 对象。第一次插入时会发生这种情况(Foo 构造函数不会抛出):

foo_list.emplace_back(new Foo(3));

切勿使用emplace_back 将原始指针插入unique_ptr 的容器中,而是使用make_unique

foo_list.emplace_back(std::make_unique<Foo>(3));

或者,如果您必须使用 C++11,则构造 unique_ptr 并插入或放置它,而不是原始指针:

foo_list.emplace_back(std::unique_ptr<Foo>(new Foo(3)));

这样,对象立即归unique_ptr所有,因此如果emplace_back内部发生异常,对象将被正确销毁。

【讨论】:

  • 谢谢,顺便说一句不错的日记!例如,我也不知道 std::tie。非常有用:)
猜你喜欢
  • 2021-06-02
  • 2015-03-24
  • 1970-01-01
  • 2017-01-12
  • 1970-01-01
  • 2015-01-15
  • 1970-01-01
  • 1970-01-01
  • 2012-02-20
相关资源
最近更新 更多