c – 静态constexpr成员的统一初始化

前端之家收集整理的这篇文章主要介绍了c – 静态constexpr成员的统一初始化前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
根据:
constexpr static data member giving undefined reference error
static constexpr类成员必须满足两个要求:
  1. template <typename Tp>
  2. struct wrapper {
  3. static constexpr Tp value{}; // 1
  4. };
  5.  
  6. template<typename Tp>
  7. constexpr Tp wrapper<Tp>::value; // 2
  8.  
  9. struct foo {
  10. };
  11.  
  12. int main() {
  13. auto const& x = wrapper<foo>::value;
  14. (void)x;
  15. }

>在类定义中初始化(因为它是constexpr)
>在类定义之外定义(因为它是静态的)

如果我改变1.统一初始化

  1. template <typename Tp>
  2. struct wrapper {
  3. static constexpr auto value = Tp{}; // uniform initialization
  4. };
  5.  
  6. template<typename Tp>
  7. constexpr Tp wrapper<Tp>::value;

编译器抱怨冲突的声明:

  1. $g++ prog.cc -Wall -Wextra -std=c++1z -pedantic
  2. prog.cc:7:31: error: conflicting declaration 'constexpr const Tp wrapper<Tp>::value' constexpr Tp wrapper<Tp>::value;
  3. prog.cc:3:29: note: prevIoUs declaration as 'constexpr const auto wrapper<Tp>::value' static constexpr auto value = Tp{};

以及缺少初始化程序:

  1. prog.cc:7:31: error: declaration of 'constexpr const auto wrapper<Tp>::value' has no initializer

正如预期的那样,删除冲突的2.定义会因链接错误而结束:

  1. In function `main': prog.cc:(.text+0x8): undefined reference to `wrapper<foo>::value'

代码示例online.

对静态constexpr类成员使用统一初始化是否可行/合法?

解决方法

这可能是我的误解,但我会认为
  1. struct wrapper {
  2. static constexpr Tp value = Tp{};
  3. };

作为统一初始化的一个例子.实际上,第一个代码示例也是统一初始化.标准本身只是要求使用大括号或赋值表达式初始化这些静态constexpr成员.正如您已经看到的那样,这很好.

问题似乎是模板上下文中auto的类型推导,我怀疑它是一个实现错误,虽然标准很大,我可能很容易错过一些东西.

如果constexpr初始化的右手大小是具有难以预先确定类型的表达式,则解决方法是使用decltype,例如,

  1. template <typename Tp>
  2. struct wrapper {
  3. static constexpr decltype(complex-init-expr) value = complex-init-expr;
  4. };
  5.  
  6. template <typename Tp>
  7. static constexpr decltype(complex-init-expr) wrapper<Tp>::value;

要么

  1. template <typename Tp>
  2. struct wrapper {
  3. typedef decltype(complex-init-expr) value_type;
  4. static constexpr value_type value = complex-init-expr;
  5. };
  6.  
  7. template <typename Tp>
  8. static constexpr typename wrapper<Tp>::value_type wrapper<Tp>::value;

猜你在找的C&C++相关文章