CNL  2.0.2 (development)
Compositional Numeric Library
hash.h
1 
2 // Copyright John McFarlane 2018.
3 // Distributed under the Boost Software License, Version 1.0.
4 // (See accompanying file ../LICENSE_1_0.txt or copy at
5 // http://www.boost.org/LICENSE_1_0.txt)
6 
7 #if !defined(CNL_IMPL_FRACTION_HASH_H)
8 #define CNL_IMPL_FRACTION_HASH_H
9 
10 #include "../../bit.h"
11 #include "../num_traits/width.h"
12 #include "canonical.h"
13 #include "definition.h"
14 
15 #include <concepts>
16 #include <functional>
17 
18 namespace std {
19  template<typename Numerator, typename Denominator>
20  struct hash<cnl::fraction<Numerator, Denominator>> {
21  // Not implemented for floating-point components.
22  // The problem (1./2.) == (2./4.) but the hashes are not equal.
23  // There is no equivalent to GCD for floating-point fractions.
24  // The solution may be to canonicalize the exponents somehow.
25  static_assert(
26  std::is_integral_v<Numerator> && std::is_integral_v<Denominator>,
27  "std::hash<cnl::fractional<T>> - T must be an integer");
28 
29  [[nodiscard]] constexpr auto operator()(
30  cnl::fraction<Numerator, Denominator> const& value) const
31  {
32  return from_canonical(cnl::_impl::canonical(value));
33  }
34 
35  private:
36  [[nodiscard]] static constexpr auto from_canonical(
38  {
39  return from_canonical_hashes(
40  hash<Numerator>{}(value.numerator), hash<Denominator>{}(value.denominator));
41  }
42 
43  [[nodiscard]] static constexpr auto from_canonical_hashes(size_t const n, size_t const d)
44  {
45  return n ^ cnl::rotl(d, cnl::_impl::width<size_t> / 2);
46  }
47  };
48 }
49 
50 #endif // CNL_IMPL_FRACTION_HASH_H
std::hash::operator()
T operator()(T... args)
cnl::fraction
numeric type represented as the fraction, numerator / denominator
Definition: definition.h:28
cnl
compositional numeric library
Definition: abort.h:15
std
STL namespace.