c++ 对类型特征的多个std::void_t偏特化进行排序的可靠方法

hujrc8aj  于 2023-06-25  发布在  其他
关注(0)|答案(2)|浏览(120)

我想对一个类型的“特性支持级别”进行分类。类型可以包含别名redblue。它可以只包含一个,两个,或者都不包含,我想用enum class对这些情况进行分类:

  1. enum class holder_type {
  2. none,
  3. red,
  4. blue,
  5. both
  6. };

这是“成员检测器习惯用法”的一个用例,我尝试使用std::void_t实现它:

  1. // primary template
  2. template <class T, class = void, class = void>
  3. struct holder_type_of
  4. : std::integral_constant<holder_type, holder_type::none> {};
  5. // substitution failure if no alias T::red exists
  6. template <class T, class Void>
  7. struct holder_type_of<T, std::void_t<typename T::red>, Void>
  8. : std::integral_constant<holder_type, holder_type::red> {};
  9. // substitution failure if no alias T::blue exists
  10. template <class T, class Void>
  11. struct holder_type_of<T, std::void_t<typename T::blue>, Void>
  12. : std::integral_constant<holder_type, holder_type::blue> {};
  13. // substitution failure if one of the aliases doesn't exist
  14. template <class T>
  15. struct holder_type_of<T, std::void_t<typename T::blue>, std::void_t<typename T::red>>
  16. : std::integral_constant<holder_type, holder_type::both> {};

然而,第一和第二部分专门化是彼此的重新定义。下面的Assert不能用clang编译,但用GCC编译成功(如所需)。

  1. static_assert(holder_type_of<red_holder>::value == holder_type::red);
  2. static_assert(holder_type_of<blue_holder>::value == holder_type::blue);
  1. <source>:36:8: error: redefinition of 'holder_type_of<T, std::void_t<typename T::blue>, Void>'
  2. struct holder_type_of<T, std::void_t<typename T::blue>, Void>
  3. ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. <source>:32:8: note: previous definition is here
  5. struct holder_type_of<T, std::void_t<typename T::red>, Void>
  6. ^

如何让我的代码在所有编译器上编译,所有Assert都通过?还有,clang在这里是不是错了,我的代码实际上应该按照标准工作?

  • 注意:这个例子是故意人为的,但也可以在实践中使用,比如将迭代器分类为随机存取/转发/等。我在尝试检测用户可以自己专门化的trait类型的成员时遇到过这个问题。
dly7yett

dly7yett1#

这看起来像CWG1980std::void_t<typename T::red>std::void_t<typename T::blue>被认为是“等价的”,因为它们都是void,所以它们是重定义,但它们在功能上不等价,因为它们可以通过替换失败来区分)。
即使你要通过使它成为一个依赖的void来修复它,比如:

  1. template<typename...> struct dependent_void { using type = void; };
  2. template<typename... T> using dependent_void_t = typename dependent_void<T...>::type;

它会使这两个部分特化:

  1. template <class T, class Void>
  2. struct holder_type_of<T, dependent_void_t<typename T::red>, Void>
  3. : std::integral_constant<holder_type, holder_type::red> {};
  4. template <class T>
  5. struct holder_type_of<T, dependent_void_t<typename T::blue>, dependent_void_t<typename T::red>>
  6. : std::integral_constant<holder_type, holder_type::both> {};

因为中间参数dependent_void_t<typename T::red>dependent_void_t<typename T::blue>是不相关的,并且不是一种或另一种方式更好。
...所以你可以翻转红色的参数,这样最后一个参数是相同的,dependent_void_t<typename T::blue>Void更好。你甚至可以回到std::void_t,因为你不再用不同的模板参数比较多个std::void_t

  1. template <class T, class Void>
  2. struct holder_type_of<T, Void, std::void_t<typename T::red>>
  3. : std::integral_constant<holder_type, holder_type::red> {};

https://godbolt.org/z/s6dPYWeao
在多种情况下,这很快就变得难以管理
基于“优先级”的SFINAE检测更容易通过函数重载完成,尽管:

  1. // Lowest priority overload
  2. template<typename T>
  3. constexpr holder_type holder_type_of_impl(...) { return holder_type::none; }
  4. template<typename T, std::void_t<typename T::red>* = nullptr>
  5. constexpr holder_type holder_type_of_impl(void*) { return holder_type::red; }
  6. template<typename T, std::void_t<typename T::blue>* = nullptr>
  7. constexpr holder_type holder_type_of_impl(long) { return holder_type::blue; }
  8. template<typename T, std::void_t<typename T::red, typename T::blue>* = nullptr>
  9. constexpr holder_type holder_type_of_impl(int) { return holder_type::both; }
  10. // Highest priority overload
  11. template <class T>
  12. struct holder_type_of
  13. : std::integral_constant<holder_type, ::holder_type_of_impl<T>(0)> {};
  14. // And if you have more than 4 priorities (probably a sign to break up your checks),
  15. // you can use this handy template
  16. template<unsigned N> struct priority : priority<N-1u> {};
  17. template<> struct priority<0> {};
  18. // Ordered from worst to best match for `f(priority<6>{})`
  19. void f(priority<0>);
  20. void f(priority<1>);
  21. void f(priority<2>);
  22. void f(priority<3>);
  23. void f(priority<4>);
  24. void f(priority<5>);
  25. void f(priority<6>);
展开查看全部
ukqbszuj

ukqbszuj2#

代码的问题是由<T, void_t<>>的几个专门化引起的,这些专门化被评估为<T, void>,因此出现了重新定义错误。
为了避免这个问题,您需要对每个专门化进行一些消歧。基本上,我认为没有办法逃避为has_type编写专门化。
然而,使用标签可以使生活变得更容易。
下面是一个方便的API示例,它反映了您的人工示例中的用法:

  1. #include <type_traits>
  2. struct S {};
  3. struct empty {};
  4. struct red_holder {
  5. using red = S;
  6. };
  7. struct blue_holder {
  8. using blue = S;
  9. };
  10. struct both_holder {
  11. using red = S;
  12. using blue = S;
  13. };
  14. enum /*class*/ holder_type {
  15. none = 0,
  16. red = 1,
  17. blue = 2,
  18. both = 3
  19. };
  20. namespace tag
  21. {
  22. struct red_t {} constexpr red{};
  23. struct blue_t{} constexpr blue{};
  24. }
  25. template <typename Tag, typename T, typename = void>
  26. struct _has : std::integral_constant<holder_type, holder_type::none>{};
  27. template <typename T>
  28. struct _has<tag::red_t, T, std::void_t<typename T::red>>
  29. : std::integral_constant<holder_type, holder_type::red> {};
  30. template <typename T>
  31. struct _has<tag::blue_t, T, std::void_t<typename T::blue>>
  32. : std::integral_constant<holder_type, holder_type::blue> {};
  33. template <typename T, typename ... Tags>
  34. constexpr holder_type has(Tags... tags) noexcept {
  35. const auto flags = (_has<Tags, T>{} | ...);
  36. return static_cast<holder_type>(flags);
  37. }
  38. template <typename T>
  39. constexpr holder_type has() noexcept {
  40. return holder_type::none;
  41. }
  42. static_assert(has<empty>() == holder_type::none);
  43. static_assert(has<red_holder>(tag::red) == holder_type::red);
  44. static_assert(has<blue_holder>(tag::blue) == holder_type::blue);
  45. static_assert(has<both_holder>(tag::red) == holder_type::red);
  46. static_assert(has<both_holder>(tag::blue) == holder_type::blue);
  47. static_assert(has<both_holder>(tag::blue, tag::red) == holder_type::both);

https://godbolt.org/z/qPax11GYf
因为(在我看来)你需要有类型->值Map,我建议使用标志。
使用标记减少了必要的样板文件。基本上,对于每个成员检测,您必须编写2个类模板(基本“假”+专门化)。
通过使用标记,您现在只需编写一个专门化。但是在我们有反射支持之前,你必须***编写它。

展开查看全部

相关问题