一个分数类

下面来看一个非常普通的例子,用来展示实现抽象数据类型的一个用户自定义类:Fraction(分数). 我们已经知道 Python 给我们提供了大量的类. 有很多可以适当地帮我们构建分数类型的数据对象.

一个分数比如cd/usr/local/hadoop@Master:/usr/local

Fraction 类的方法应该能够让 Fraction 对象可以像其他的数值那样进行计算. 我们需要可以进行分数之间的 加, 减, 乘, 和 除 运算. 更进一步, 所有的方法应该返回最简分数.

在 Python中, 我们定义一个类的名字还有一些方法,类似于定义一个函数,例如,

class Fraction:#the methods go here

提供了给我们定义方法的框架.第一个方法是所有类都要提供的构造器. 构造函数定义了类创建的方式. 要创建分数对象, 我们需要提供两部分的数据:分子和分母. 在 Python中, 构造函数使用 __init__ (两条下划线包围 init) ,如下所示:

Listing 2

class Fraction:def __init__(self,top,bottom):self.num = topself.den = bottom

注意到参数列表含有三个参数: (selftopbottom). self 是一个引用对象自身的特殊的参数. 它通常作为第一个参数; 但是, 它从不在调用的时候传值. 之前已经讲过,分数包含两部分(分子和分母). 记号 self.num 在构造函数中被定义为 fraction 对象具有一个叫num 的内部数据对象. 同理, self.den 也是类似的目的.

要实现 Fraction 类, 我们需要调用构造函数. 接着通过类名传递参数 (注意到我们从不直接调用__init__). 例如:

myfraction = Fraction(3,5)

创建一个分数对象 myfraction 代表分数3/5 .

接着要做的事情就是给抽象数据类型实现方法. 首先, 意识到当我们要输出一个 Fraction 对象.

>>> myf = Fraction(3,5)
>>> print(myf)
<__main__.Fraction instance at 0x409b1acc>

fraction 对象, myf, 并不知道怎样响应输出操作.  print 函数需要对象转换为可输出的字符串格式,这样才能输出. 唯一的选择 myf 必须显示变量实际的地址引用(自身的地址). 这不是我们想要的.

有两种解决问题的办法. 一种是定义一种称为 show 的方法,可以将Fraction 对象作为一个字符串的形式打印. 我们可以实现如 Listing 3所示.假如我们按照前面讲的创建 Fraction 对象, 我们可以让它输出自身, 换句话说, 打印自身按照适当的格式. 不幸的是, 这通常不起作用. 为了使输出工作正常, 我们必须告诉 Fraction 类怎样将自身转换为字符串的格式.

Listing 3

def show(self):print(self.num,"/",self.den)
>>> myf = Fraction(3,5)
>>> myf.show()
3 / 5
>>> print(myf)
<__main__.Fraction instance at 0x40bce9ac>

在 Python, 所有的类都提供但不是都适用的标准的方法. 其中之一, __str__,就是一个将对象转换为字符串的方法. 这个方法默认的实现是用来以字符串格式返回类实例的地址. 我们必须为这个方法提供一个“更好的”实现. 我们说这个新的方法重载前面的, 或者说重新定义了方法的行为.

要实现这个,我们简单地定义一个名叫 __str__ 的方法并给出实现 如Listing 4. 这个定义除了使用特殊参数 self以外不需要其他的信息. 注意函数中的不同的实现办法.

Listing 4

def __str__(self):return str(self.num)+"/"+str(self.den)
>>> myf = Fraction(3,5)
>>> print(myf)
3/5
>>> print("I ate", myf, "of the pizza")
I ate 3/5 of the pizza
>>> myf.__str__()
'3/5'
>>> str(myf)
'3/5'
>>>

我们可以为我们的新 Fraction 类覆盖很多其他的方法. 其中一些最重要的是一些基础的算术运算操作. 我们可以创建两种 Fraction 对象,同时使用“+” 符号将它们相加 . 这时, 如果我们使两分数相加, 我们得到:

>>> f1 = Fraction(1,4)
>>> f2 = Fraction(1,2)
>>> f1+f2
Traceback (most recent call last):File "<pyshell#173>", line 1, in -toplevel-f1+f2
TypeError: unsupported operand type(s) for +:'instance' and 'instance'

如果你仔细观察错误信息, 你将发现问题是: “+” 操作符不能理解Fraction 操作.

我们可以通过给 Fraction 类提供重载的加法函数来实现. 在 Python, 这种方法称为 __add__ 同时需要两个参数. 第一个参数, self,  第二个参数是另一个操作数. 例如,

f1.__add__(f2)

当 Fraction  f1 加 Fraction f2. 可以写成标准的形式:f1+f2.

两个分数必须有相同的分母才能直接相加. 使它们分母相同最简单的方法是通分: ,具体实现如 Listing 5. 加法函数返回了一个新的 Fraction 对象.

Listing 5

def __add__(self,otherfraction):newnum = self.num * otherfraction.den + self.den*otherfraction.numnewden = self.den * otherfraction.denreturn Fraction(newnum,newden)

>>> f1=Fraction(1,4)
>>> f2=Fraction(1,2)
>>> f3=f1+f2
>>> print(f3)
6/8
>>>

上面的加法函数看起来实现了我们期望的, 但是还可以更完美. 注意到 6/8 是正确的结果,但是却不是以 “最简项” 的形式展示的. 最好的表达式为3/4. 为了使我们的结果为最简项的形式, 我们需要一个辅助函数才化简分数. 这个函数可以求出最大公约数, 或者称为 GCD. 可以通过分子和分母的最大公约数来达到化简分数的目的.

计算最大公约数最著名的算法要数 Euclid算法,原理我就不详细指明了,很简单。实现如下:

>>> def gcd(m, n):while m % n != 0:oldm = moldn = nm = oldnn = oldm % oldnreturn n>>> print gcd(20, 10)
10

这样我们就可以化简任何的分数了,代码如下: (Listing 6).

Listing 6

def __add__(self,otherfraction):newnum = self.num*otherfraction.den + self.den*otherfraction.numnewden = self.den * otherfraction.dencommon = gcd(newnum,newden)return Fraction(newnum//common,newden//common)

>>> f1=Fraction(1,4)
>>> f2=Fraction(1,2)
>>> f3=f1+f2
>>> print(f3)
3/4

我们的 Fraction 对象现在有两个非常重要的方法,如上图所示. 一些需要新增进我们的实例类 Fraction 的方法是:允许两个分数进行比较. 假如我们有两个 Fraction 对象, f1 和f2f1==f2 将得到True 假如他们指向同一个对象. 即使分子分母都相同,但是不满足条件依然将不相等. 这被称为 shallow equality (如下图).

我们可以创建 deep equality (如上图)–通过值相等来判断, 不同于引用–通过覆盖 __eq__ 方法.  __eq__ 是另一个存在于所有类中标准方法. __eq__ 方法比较两个对象当值相等的时候返回 True ,否则返回 False.

在 Fraction 类中, 我们实现了 __eq__ 方法通过常规比较方法来比较分数 (see Listing 7). 值得注意的是还有其他的方法可以覆盖. 例如,  __le__ 方法提供了小于等于功能.

Listing 7

def __eq__(self, other):firstnum = self.num * other.densecondnum = other.num * self.denreturn firstnum == secondnum

完整的 Fraction 类的代码如下所示:

def gcd(m,n):while m%n != 0:oldm = moldn = nm = oldnn = oldm%oldnreturn nclass Fraction:def __init__(self,top,bottom):self.num = topself.den = bottomdef __str__(self):return str(self.num)+"/"+str(self.den)def show(self):print(self.num,"/",self.den)def __add__(self,otherfraction):newnum = self.num*otherfraction.den + \self.den*otherfraction.numnewden = self.den * otherfraction.dencommon = gcd(newnum,newden)return Fraction(newnum//common,newden//common)def __eq__(self, other):firstnum = self.num * other.densecondnum = other.num * self.denreturn firstnum == secondnumx = Fraction(1,2)
y = Fraction(2,3)
print(x+y)
print(x == y)

运行结果:

7/6False

Python数据结构与算法--面向对象相关推荐

  1. python leetcode_leetcode 介绍和 python 数据结构与算法学习资料

    for (刚入门的编程)的高中 or 大学生 leetcode 介绍 leetcode 可以说是 cs 最核心的一门"课程"了,虽然不是大学开设的,但基本上每一个现代的高水平的程序 ...

  2. Python数据结构与算法(1.6)——Python基础之类与模块化

    Python数据结构与算法(1.6)--Python基础之类与模块化 0. 学习目标 1. 面向对象编程:类 1.1 面向对象编程的基本概念 1.1.1 多态 1.1.2 封装 1.1.3 继承 1. ...

  3. Python天天美味(32) - python数据结构与算法之堆排序

    1. 选择排序 选择排序原理是先选出最小的数,与第一个数交换,然后从第二个数开始再选择最小的数与第二个数交换,-- def selection_sort(data):     for i in ran ...

  4. python数据结构与算法13_python 数据结构与算法 (13)

    python 数据结构与算法 (13) 选择排序 (Selection sort) 是? 种简单直观的排序算法. 它的? 作原理如 下.? 先在未排序序列中找到最?(?)元素, 存放到排序序列的起始位 ...

  5. Python数据结构与算法(二)栈和队列

    本系列总结了python常用的数据结构和算法,以及一些编程实现. 参考书籍:<数据结构与算法 Python语言实现> [美]Michael T.Goodrich, Roberto Tama ...

  6. Python数据结构与算法(一)列表和元组

    本系列总结了python常用的数据结构和算法,以及一些编程实现. 参考书籍:<数据结构与算法 Python语言实现> [美]Michael T.Goodrich, Roberto Tama ...

  7. Python 数据结构与算法——快排

    Python 数据结构与算法--选取算法(TopK) 如果说快速选取法所代表的是剪枝式的遍历操作--在递归树中找出一条通往第 k<script type="math/tex" ...

  8. Python 数据结构与算法——从二分图到寻找最大排列(Maximum Permutation)

    假设现在有 8 位有着特殊癖好的人去买票看电影,其中有一部分人得到了自己喜欢的座位,但大多数人并不满意.现在的问题是,如果这些人各自都有自己喜欢的座位(喜欢的座位有重叠,这是进行最大排列的前提,否则无 ...

  9. Python数据结构与算法(4.1)——递归

    Python数据结构与算法(4.1)--递归 0. 学习目标 1 递归 1.1 递归的基本概念 1.2 递归的重要性 1.3 递归三原则 1.4 递归的应用 2 递归示例 2.1 列表求和 2.2 汉 ...

  10. Python数据结构与算法(3.5)——双端队列

    Python数据结构与算法(3.5)--双端队列 0. 学习目标 1. 双端队列的基本概念 1.1 双端队列的基本概念 1.2 双端队列抽象数据类型 2. 双端队列的实现 2.1 顺序双端队列的实现 ...

最新文章

  1. 2019 Multi-University Training Contest 1 - 1004 - Vacation - 二分 - 思维
  2. CAN笔记(5) 协议标准规格
  3. 如何使用Git上传项目代码到github
  4. dataframe 输出标题_【学界】第八章:Python代码之数据输出、调参与算法总结
  5. Yii2 使用 .env 来配置项目环境变量
  6. 智能指针auto_ptr源码
  7. yagmail发送附件
  8. 计算机图形学入门:什么是光线追踪?
  9. 别再说互联网寒冬了,为何不从自己身上找找原因呢
  10. 基于RSA+AES 软件授权License
  11. Java之各平台快递对接
  12. 函数概念与特性——反函数、复合函数
  13. MySQL数据库优化高频面试题(最新版)
  14. macOS上使用Openconnect代替Cisco Anyconnect
  15. java 数学测试_Java实现小学数学练习
  16. 会员管理系统与微信连接能实现哪些功能?
  17. svg-captcha验证码识别,成功率100%
  18. Python+Vue计算机毕业设计框架的综合性电商6si2i(程序+LW+源码+部署)
  19. 详解流编辑器 sed 和 编程语言 awk
  20. 静态时序分析(STA)——建立约束

热门文章

  1. Rust : 用rust实现Diffe-Hellman算法
  2. 持续图片滚动字幕html,使用JavaScript实现连续滚动字幕效果的方法
  3. 毕设题目:Matlab语音去噪
  4. 全自动与半自动手表的区别_半自动和全自动的区别在哪里?
  5. mysql数据库回滚日志_Mysql数据库慢查询日志的使用
  6. 怎么将php文件改成web的servlet文件_遇到喜欢的网站怎么才能高效收藏整理
  7. html获取视频时长,js获取本地视频时间长度
  8. python:批量修改文件夹下所有图片名字,改为文件夹名字的一部分+.jpg
  9. 斐波那契数列的Python简单实现
  10. c语言连接mysql_0基础掌握Django框架(14)MySQL相关软件