Underscores are now permitted when creating a :class:`Fraction` instance
from a string, following :PEP:`515` rules.
+ .. versionchanged:: 3.11
+ :class:`Fraction` implements ``__int__`` now to satisfy
+ ``typing.SupportsInt`` instance checks.
+
.. attribute:: numerator
Numerator of the Fraction in lowest term.
fractions
---------
-Support :PEP:`515`-style initialization of :class:`~fractions.Fraction` from
-string. (Contributed by Sergey B Kirpichev in :issue:`44258`.)
+* Support :PEP:`515`-style initialization of :class:`~fractions.Fraction` from
+ string. (Contributed by Sergey B Kirpichev in :issue:`44258`.)
+
+* :class:`~fractions.Fraction` now implements an ``__int__`` method, so
+ that an ``isinstance(some_fraction, typing.SupportsInt)`` check passes.
+ (Contributed by Mark Dickinson in :issue:`44547`.)
math
"""abs(a)"""
return Fraction(abs(a._numerator), a._denominator, _normalize=False)
+ def __int__(a, _index=operator.index):
+ """int(a)"""
+ if a._numerator < 0:
+ return _index(-(-a._numerator // a._denominator))
+ else:
+ return _index(a._numerator // a._denominator)
+
def __trunc__(a):
- """trunc(a)"""
+ """math.trunc(a)"""
if a._numerator < 0:
return -(-a._numerator // a._denominator)
else:
import fractions
import functools
import sys
+import typing
import unittest
from copy import copy, deepcopy
import pickle
self.assertTypedEquals(0.1+0j, complex(F(1,10)))
+ def testSupportsInt(self):
+ # See bpo-44547.
+ f = F(3, 2)
+ self.assertIsInstance(f, typing.SupportsInt)
+ self.assertEqual(int(f), 1)
+ self.assertEqual(type(int(f)), int)
+
+ def testIntGuaranteesIntReturn(self):
+ # Check that int(some_fraction) gives a result of exact type `int`
+ # even if the fraction is using some other Integral type for its
+ # numerator and denominator.
+
+ class CustomInt(int):
+ """
+ Subclass of int with just enough machinery to convince the Fraction
+ constructor to produce something with CustomInt numerator and
+ denominator.
+ """
+
+ @property
+ def numerator(self):
+ return self
+
+ @property
+ def denominator(self):
+ return CustomInt(1)
+
+ def __mul__(self, other):
+ return CustomInt(int(self) * int(other))
+
+ def __floordiv__(self, other):
+ return CustomInt(int(self) // int(other))
+
+ f = F(CustomInt(13), CustomInt(5))
+
+ self.assertIsInstance(f.numerator, CustomInt)
+ self.assertIsInstance(f.denominator, CustomInt)
+ self.assertIsInstance(f, typing.SupportsInt)
+ self.assertEqual(int(f), 2)
+ self.assertEqual(type(int(f)), int)
+
def testBoolGuarateesBoolReturn(self):
# Ensure that __bool__ is used on numerator which guarantees a bool
# return. See also bpo-39274.