具有不同概念的模板类专业化会导致重新定义错误

ofo*_*ofo 6 c++ template-specialization c++-concepts c++20

我正在使用std::hash一个Hashable概念编写一些代码。然而,即使我没有模糊的实例,我也无法用两个不同的概念来定义专业化。

#include <ranges>
#include <concepts>
#include <functional>

namespace ranges = std::ranges;

template <typename T>
concept Hashable = requires(const T& t) {
  { std::hash<T>{}(t) } -> std::convertible_to<size_t>;
};

template <typename T>
concept HashableWithMember = requires(const T& t) {
  { t.Hash() } -> std::convertible_to<size_t>;
};

template <typename R>
concept HashableRange = ranges::range<R> && Hashable<ranges::range_value_t<R>>;

template <HashableWithMember T>
struct std::hash<T> {
  size_t operator()(const T& t) const { return t.Hash(); }
};

template <HashableRange R>
struct std::hash<R> {
  size_t operator()(const R& r) const {
    return 0;
  }
};

template <Hashable T>
static size_t Hash(const T& t) {
  return std::hash<T>{}(t);
}
Run Code Online (Sandbox Code Playgroud)
<source>:26:13: error: redefinition of 'struct std::hash<_Tp>'
   26 | struct std::hash<R> {
      |             ^~~~~~~
<source>:21:13: note: previous definition of 'struct std::hash<_Tp>'
   21 | struct std::hash<T> {
      |             ^~~~~~~
Run Code Online (Sandbox Code Playgroud)

https://godbolt.org/z/1roj1qMbs

如果有一个类既是 aHashableRange又是 a HashableWithMember,我会理解这个错误,但没有。为什么这不起作用?

Fed*_*dor 2

这是已知的 GCC 错误[concepts] redefinition error when using constrained structure template inside namespacehttps://gcc.gnu.org/bugzilla/show_bug.cgi ?id=92944

一个更简单的例子如下:

#include <concepts>
#include <functional>

template <std::convertible_to<int> T>
struct std::hash<T> {};

template <std::convertible_to<float> U>
struct std::hash<U> {};
Run Code Online (Sandbox Code Playgroud)

Clang 和 MSVC 都可以,bug 仅限于 GCC。演示: https: //gcc.godbolt.org/z/Warhfaf7q