C++编译器何时推断 no,除了方法?

When does a C++ compiler infer noexcept for a method?

本文关键字:方法 no 编译器 何时推 C++      更新时间:2023-10-16

我只是注意到我的一个std::vector<Foo>在调整大小时正在复制而不是移动其元素 - 即使Foo有一个移动 ctor:

class Foo {
// ...
Foo(Foo&& other) : id_(other.id_), ptr_(other.ptr_), flag(other.flag)
{
other.flag = false;
};
// ...
int   id_; 
void* ptr_; 
bool  flag;
}

然后我读到:

在 std::vector 上调整大小不会调用移动构造函数

这提醒了我,只有当元素的移动 ctor 被声明为noexcept时,std::vector才会使用移动构造。当我添加noexcept时,移动ctor被调用。

我的问题是:为什么给定移动 ctor 的代码,编译器不确定它是noexcept的?我的意思是,它可以知道一个事实,即不能抛出异常。另外,标准是否不允许推断noexcept,或者不仅仅是由我的特定编译器完成的?

我在GNU/Linux上使用GCC 5.4.0。

tl;dr:不允许编译器推断noexcept

为什么,给定移动 ctor 的代码,编译器不确定它是 noexexcept ?

因为 noexcept 规范是根据声明确定的 - 而不是定义。这类似于 const 规范的工作方式。不允许编译器将函数确定为 const,即使其实现不修改任何成员。

是推断 否,除非标准不允许

据我了解,是的:

[规格除外] ...中缺少异常规范 除析构函数 (12.4( 或释放函数 (3.7.4.2( 的函数声明符之外的函数声明符表示 异常规范,即所有类型的集合。

推断所有类型的集合以外的内容将与此规则相矛盾。当然,当编译器可以证明不能抛出异常时,它可以在 as-if 规则下优化任何堆栈展开代码,但这种优化不会影响 SFINAE 自省。


已经有关于引入noexcept(auto)的可能性的讨论,这将是让编译器推断noexcept 规范的明确方法。