1.3 fractions模块

时间:2022-04-09 13:33:48

数学世界中,浮点数还可以用分数形式展示,不可约简的分数形式往往更简洁直观。

 问题来了,Python中如何输出不可约简的分数形式呢?
 答案:用Fraction类来实现。这个类属于标准库的fractions模块。按照国际惯例,用之前需要先导入模块:from fractions import Fraction
 
 下面重点介绍fractions的核心:Fraction类。
 
  1,该类的构造方法:class fractions.Fraction(numerator = 0,denominator = 1)
    Fraction在实例化的时候,需要的参数是2个,numerator = 0(分子)、denominator = 1(分母),
      这两个参数都指定了默认参数且这两个参数必须同时是int类型或者numbers.Rational类型,否则会抛出类型错误;
      如果在实例化时输入的分母为0,会抛出异常ZeropisionError;
      如果在实例化时只输入一个参数值的时候,这个参数值会赋给numerator参数作为分子,分母就是默认值1。
 
  2,  Farction在实例化时,能输入的参数类型有如下五种:int、float、str、Decimal、Fraction
      如果在实例化时只提供一个参数,则可以用上述五种类型进行初始化;
      如果使用字符串进行初始化时,fractions模块使用内置的正则表达式进行匹配;
      如果使用浮点数或者Decimal进行初始化时,fractions模块会在内部调用as_integer_ratio()函数进行处理

国际惯例的时刻到了,介绍完一个知识点,来点练习加强一下学习效果

 >>> from fractions import Fraction

  (1)中规中矩地用两个参数来实例化Fraction类

 >>> Fraction(2,-4)
 Fraction(-1, 2)           # 1:Fraction()会把实例化参数约分后输出;2:会把分母上的负号“-”转到分子上
 >>>

  (2)只用一个整数来实例化Fraction类

 >>> Fraction(3)

 Fraction(3, 1)            #看到没,实例化时输入的参数3是赋值给分子,而分母是默认参数1
 >>>

  (3)只用一个字符串实例化Fraction类

 >>> Fraction(')          

 Fraction(4, 1)

 >>>
 >>> Farction('s')
 Traceback (most recent call last):
 File "<pyshell#10>", line 1, in <module>
 Farction('s')
 NameError: name 'Farction' is not defined
 >>>

 >>> Fraction('4.1')
 Fraction(41, 10)
 >>>

 #如果使用字符串进行初始化时,fractions模块使用内置的正则表达式进行匹配

 >>> Fraction('\t\n 1.2 \t\n')
 Fraction(6, 5)     #正则表达式把字符串两侧的空格、Tab、换行符都给剔除了

 #脑子中突然闪现出一个调皮的想法:我把空格、Tab、换行符放在数字中间会如何

 >>> Fraction('1, \t\n 2')
 Traceback (most recent call last):
 File "<pyshell#33>", line 1, in <module>
 Fraction('1, \t\n 2')
 File "C:\Users\dddd\AppData\Local\Programs\Python\Python35\lib\fractions.py", line 146, in __new__numerator)
 ValueError: Invalid literal for Fraction: '1, \t\n 2'         # 啊!好爽!!

 #我又要调皮了,如果一个浮点数的只有小数位会怎么样?

 >>> Fraction('-.25')

 Fraction(-1, 4)      #这样也行!!!

     
 #最后的最后,怎么能忽略了数学中的科学计数法表示的数呢

 >>> Fraction('1.2*e-6')
 Traceback (most recent call last):
 File "<pyshell#40>", line 1, in <module>
 Fraction('1.2*e-6')
 File "C:\Users\dddd\AppData\Local\Programs\Python\Python35\lib\fractions.py", line 146, in __new__numerator)
 ValueError: Invalid literal for Fraction: '1.2*e-6'

 >>> Fraction('1e-6')
 Fraction(1, 1000000)

 >>> Fraction('e-6')
 Traceback (most recent call last):
 File "<pyshell#42>", line 1, in <module>
 Fraction('e-6')
 File "C:\Users\dddd\AppData\Local\Programs\Python\Python35\lib\fractions.py", line 146, in __new__numerator)
 ValueError: Invalid literal for Fraction: 'e-6'
 >>>

 #可见并不是什么字符串都可以作为Fraction的参数的,需要是整形或浮点型转化成的浮点数

  (4)以分数作为参数实例化Faction类

 >>> Fraction(1/7)

 Fraction(2573485501354569, 18014398509481984)   # 我操!什么鬼?

 >>> Fraction('1/7')
 Fraction(1, 7)
 >>>

 >>> Fraction('-1/7')

 Fraction(-1, 7)
 >>> Fraction('1/-7')
 Traceback (most recent call last):
 File "<pyshell#16>", line 1, in <module>
 Fraction('1/-7')
 File "C:\Users\dddd\AppData\Local\Programs\Python\Python35\lib\fractions.py", line 146, in __new__numerator)
 ValueError: Invalid literal for Fraction: '1/-7'
 #不知道为什么每次看到报错都会有心痛的感觉

 >>> Fraction('1/(-7)')
 Traceback (most recent call last):
 File "<pyshell#19>", line 1, in <module>
 Fraction('1/(-7)')
 File "C:\Users\dddd\AppData\Local\Programs\Python\Python35\lib\fractions.py", line 146, in __new__numerator)
 ValueError: Invalid literal for Fraction: '1/(-7)'
 #看来用'a/b'的形式的参数实例化Fraction时,负号'-'只能加在a前边了
 >>>

  (5)有时候将浮点数或者Decimal作为Fraction实例的初始化数据可能会遇到舍入误差(二进制的浮点数是不精确的)的问题,如调用Fraction(1.2)时不返回Fraction(12, 10)的。这时Fraction类提供了一个实例函数limit_denominator()  来减小这种舍入误差。

 >>> Fraction('1/7').limit_denominator(max_denominator=100)
 Fraction(1, 7)
 >>> Fraction(1/7).limit_denominator(max_denominator=100)
 Fraction(1, 7)
 >>> Fraction(1/7)
 Fraction(2573485501354569, 18014398509481984)
 >>>
 >>> Fraction(1.2)
 Fraction(5404319552844595, 4503599627370496)
 >>> Fraction(1.2).limit_denominator(max_denominator=100)
 Fraction(6, 5)
 >>>

3,由于Fraction类继承了numbers.Rational类并且重载了该类多有的方法,所以其实例可以直接进行一些算数运算、关系运算等。

   先实例化两个Fraction对象   x = Fraction(1,2)   y = Fraction(1,3)

  (1)+

 >>> x + y
 Fraction(5, 6)

  (2)-

 >>> x - y
 Fraction(1, 6)

  (3)*

 >>> x * y
 Fraction(1, 6)

  (4)/

 >>> x / y
 Fraction(3, 2)

  (5)用Python内置函数abs()处理一下呢

 >>> x*(-1)

 Fraction(-1, 2)
 >>> abs(x*(-1))         #内置函数能正常处理哦
 Fraction(1, 2)

 内置函数能处理,math模块中的函数肯定也能处理咯,试一下

 >>> import math

 >>> math.ceil(x)

 1
 >>> math.floor(x)
 0

关于fraction模块就介绍这么多吧,其实我目前还在自学阶段,没有什么实践的机会,等将来用到了再多去多了解一些。