Commit 29bb227a authored by Tim Peters's avatar Tim Peters Committed by GitHub

Add a minor `Fraction.__hash__()` optimization (GH-15313)

* Add a minor `Fraction.__hash__` optimization that got lost in the shuffle.

Document the optimizations.
parent 0567786d
...@@ -564,10 +564,25 @@ class Fraction(numbers.Rational): ...@@ -564,10 +564,25 @@ class Fraction(numbers.Rational):
try: try:
dinv = pow(self._denominator, -1, _PyHASH_MODULUS) dinv = pow(self._denominator, -1, _PyHASH_MODULUS)
except ValueError: except ValueError:
# ValueError means there is no modular inverse # ValueError means there is no modular inverse.
hash_ = _PyHASH_INF hash_ = _PyHASH_INF
else: else:
hash_ = hash(abs(self._numerator)) * dinv % _PyHASH_MODULUS # The general algorithm now specifies that the absolute value of
# the hash is
# (|N| * dinv) % P
# where N is self._numerator and P is _PyHASH_MODULUS. That's
# optimized here in two ways: first, for a non-negative int i,
# hash(i) == i % P, but the int hash implementation doesn't need
# to divide, and is faster than doing % P explicitly. So we do
# hash(|N| * dinv)
# instead. Second, N is unbounded, so its product with dinv may
# be arbitrarily expensive to compute. The final answer is the
# same if we use the bounded |N| % P instead, which can again
# be done with an int hash() call. If 0 <= i < P, hash(i) == i,
# so this nested hash() call wastes a bit of time making a
# redundant copy when |N| < P, but can save an arbitrarily large
# amount of computation for large |N|.
hash_ = hash(hash(abs(self._numerator)) * dinv)
result = hash_ if self._numerator >= 0 else -hash_ result = hash_ if self._numerator >= 0 else -hash_
return -2 if result == -1 else result return -2 if result == -1 else result
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment