我可以只为特定的专业化定义静态constexpr数据成员吗

Can I define a static constexpr data member only for certain specializations?

本文关键字:静态 定义 constexpr 数据成员 专业化 我可以      更新时间:2023-10-16

以下程序运行良好。

#include <iostream>
#include <type_traits>
template <typename DummyT = void>
struct wrapper
{
  static_assert(std::is_same<void, DummyT>::value, "Only void, please");
  static constexpr char text[] = "some string constant";
};
template <typename DummyT>
constexpr char wrapper<DummyT>::text[];
int
main()
{
  std::cout << wrapper<>::text << 'n';
}

然而,当我只为wrapper<void>定义wrapper::text时,

template <>
constexpr char wrapper<void>::text[];

那么GCC 5.3.0给了我这个链接器错误

/tmp/ccnGx3EP.o: In function `main':
main.cxx:(.text+0x5): undefined reference to `wrapper<void>::text'
collect2: error: ld returned 1 exit status

Clang 3.7.1给了我这个错误。

main.cxx:12:31: error: declaration of constexpr static data member 'text' requires an initializer
constexpr char wrapper<void>::text[];
                              ^
1 error generated.

我想知道为什么仅仅为实际使用的专业化提供一个定义是不够的。这并不是说它非常有用,因为static constexpr成员必须在class定义中初始化,所以我无论如何都不能将其专门化到定义中,但我可能希望不定义它。

轻松:

template <typename DummyT = void>
struct wrapper
{
  static_assert(std::is_same<void, DummyT>::value, "Only void, please");
};
template <>
struct wrapper<void>
{
  static constexpr char text[] = "some string constant";
};
constexpr char wrapper<void>::text[];

wrapper<>wrapper<void>都将工作,而static_assert的任何其他参数都将失败。