Python 进阶_OOP 面向对象编程_组合与继承
#目录
前言
我们定义一个类是希望能够把类当成模块来使用,并把类嵌入到我们的应用代码中,与其他的数据类型、逻辑执行流结合使用。一般来说我们可以使用两种方法在代码中利用类,那就是组合和派生。
组合
组合: 就是将不同的类混合并加入到其他类中,来 增加类的功能 / 提高代码的重用性 / 易于维护(对类的修改会直接反应到整个应用中) 。我们可以实例化一个更大的对象,同时还可以添加一些实例属性和实例方法的实现来丰富实例对象的功能。
In [1]: class NewAddrBookEntry(object): # 定义一个类
...: """New address book entry class."""
...: def __init__(self, name, phone): # 定义构造器
...: self.name = Name(name) # 创建 Name 对象
...: self.phone = Phone(phone) # 创建 Phone 对象
...: print "Created the instance for ", self.name
上述的 class NewAddrBookEntry 有它自身和其他类 Name/Phone 组合而成, 可以实现一些更为复杂的功能。
派生
当我们希望较小的类是较大的类的组件时,组合是一个很好的处理方式。但当我们希望 相同的类却具有一些不同的功能时 派生就是最好的处理方式。这也是面向对象编程最强大的功能之一 —— 使用一个已经定义好的类,扩展它的功能或者对其进行修改生成一个新的类,但却不会对原来的类造成影响。
子类会从基类继承他们的任何属性(数据和方法),这种派生是可以继承多代的,且可以同时继承多个基类。
语法:
class SubClassName(ParentClass1[, ParentClass2, ...]):
class_suite
EXAMPLE:
In [8]: class Parent(object):
...: def __init__(self):
...: print "created an instance of: ", self.__class__.__name__
...:
...:
In [9]: class Child(Parent):
...: pass
...:
In [10]: c = Child()
created an instance of: Child
In [14]: p = Parent()
created an instance of: Parent
类 Child 的实例对象 c 并没有定义 __init__()
构造器,但仍然执行了 print 语句,这是因为 Child 从 Parent 继承了其构造器。
通过继承来覆盖(重载)方法
当我们派生一个子类,但同时希望相同的方法能在子类实现不同的功能,这时我们需要使用方法的 重载。使子类的方法能够覆盖父类的同名方法。
In [17]: class Parent(object):
...: def func(self):
...: print "This is Parent."
...:
In [18]: class Child(Parent):
...: def func(self):
...: print "This is Child."
...:
...:
In [19]: p = Parent()
In [20]: p.func()
This is Parent.
In [21]: c = Child()
In [22]: c.func()
This is Child.
这里子类 Child 重载了父类 Parent 的 func() 方法,实现了不同的功能。
但仍然有些场合需要我们即能使用子类的重载方法的同时,也要求我们重现父类方法的功能。那么我们可以调用那个被我们覆盖的父类方法吗?
答案是肯定的。
In [23]: Parent.func(c)
This is Parent.
我们可以通过 ParentClassName.functionName(object)
的方式来重现父类所被覆盖的方法。当然我们还有其他的方式可以实现这个效果,EG. 在子类的重载方法里显式的调用父类的同名方法:
In [24]: class Child(Parent):
...: def func(self):
...: print "This is Child."
...: Parent.func(self)
In [25]: c = Child()
In [26]: c.func()
This is Child.
This is Parent.
两种方式本质上是相同的,都是通过 父类名结合句点表达式 来实现对父类方法的调用,而且需要注意的是,在调用的时候必须传递一个实例对象给 func(),否则会触发参数不匹配的语法错误。
还有一个更好的实现方式就是子类使用 super() 内置函数:
In [27]: class Child(Parent):
...: def func(self):
...: print "This is Child."
...: super(Child, self).func()
...:
In [28]: c = Child()
In [29]: c.func()
This is Child.
This is Parent.
super() 内置函数不仅能自动的找到父类方法,并且还是自动的为父类方法传入 self 参数来实现实例方法的绑定。
最常用的重载场景(实例方法的重载)
最常用的重载场景莫过于 重载父类的构造器 了。
在上述的例子可以看出,当我们在子类中没有重载构造器的时候,会自动的调用父类的构造器。这很明显是不符合我们的需求的,因为我们常常需要在子类中定义一些新的成员属性。但是问题是:当我们为了初始化子类中新的成员属性时,不可避免的需要重复的编写初始化从父类中继承而来的属性的代码。 这也不符合代码重用的原则,所以我们一般会采用 重载构造器(init()) + 重现父类构造器(super()) 的方式来解决这个问题。
In [32]: class Parent(object):
...: def __init__(self, name, age):
...: self.name = name
...: self.age = age
...: print "This is Parent."
...:
In [33]: class Child(Parent):
...: def __init__(self, name, age, sex): # 初始化子类实例对象的属性
...: super(Child, self).__init__(name, age) # 初始化父类实例对象的属性
...: self.sex = sex
...: print "This is Child."
...:
In [35]: p = Parent("fanguiju", "24")
This is Parent.
In [36]: c = Child("fanguiju", "24", "man")
This is Parent.
This is Child.
In [37]: c.name
Out[37]: 'fanguiju'
In [38]: c.age
Out[38]: '24'
In [39]: c.sex
Out[39]: 'man'
一般而言,我们会在子类的构造器中首先调用父类的构造器,当然这并不是强制的。只是为了我们能够在执行子类构造器的代码之前首先完成对父类属性的初始化,防止在调用从父类继承而来的属性时仍未初始化的问题出现。
使用 super() 内置函数的漂亮之处在于,我们不需要明确的给出父类的名字,交由解析器去自动的找到该子类的父类,并自动的传入 self 参数来完成绑定。这样能够让代码具有更高的灵活性,我们只需要改变子类的定义语句,就可以改变类的继承关系。
从标准类中派生(类方法的重载)
不可变数据类型的派生:定义一个精度为 2 的浮点数据类型
派生不可变标准类,经常需要重载类方法,而类方法的重载一般是重载 __new__()
,也就是所谓的 真·构造器 。
class RoundFloat(float):
def __new__(cls, value):
return float.__new__(cls, round(value, 2)) # 将类对象 float 传入参数 cls
In [44]: RoundFloat(1.1111111)
Out[44]: 1.11
真·构造器会自动的将类对象 RoundFloat 传入 cls 参数,类似于构造器__init__(self)
。
可变数据类型的派生:定义一个有序的字典数据类型
可变数据类型的派生可能不需要使用 构造器 或者 真·构造器 也能够实现。
class SortedDict(dict):
def keys(self):
return sorted(super(SortedDict, self).keys())
In [47]: for x in SortedDict((("name", "fanguiju"), ("age", 24), ("sex", "man"))):
...: print x
...:
age
name
sex
通过 SortedDict 生成的字典按照字母的顺序排序。
需要注意的是: 在 Python 2.2 之后将类和类型合并了,所以所有的数据类型都是一个类,反之我们定义了一个类也相当于定义了一个新的类型。
Python 进阶_OOP 面向对象编程_组合与继承的更多相关文章
- Python 进阶_OOP 面向对象编程_类和继承
目录 目录 类 最简单的类 类方法 构造器 __init__ 创建一个类 实例化一个对象 调用实例的方法和属性 创建子类 使用 super 来调用父类的构造器 实例化子类对象 调用子类的属性和方法 类 ...
- Python 进阶_OOP 面向对象编程_实例属性和方法
目录 目录 构造器和解构器 构造器 __init__ 真构造器 __new__ 解构器 __del__ 实例方法 Python 中的 抽象方法 实例属性 查看实例属性 实例属性和类属性的区别 访问不可 ...
- Python 进阶_OOP 面向对象编程_类属性和方法
目录 目录 类属性 调用类属性 查看类属性 特殊的类属性 类方法 真构造器 __new__ 类属性 在理解类属性之前要先搞清楚 实例属性 和 函数属性 之间的区别: 1. 实例属性:指的是实例化类对象 ...
- Python 进阶_OOP 面向对象编程_静态方法和类方法
目录 目录 静态方法 类方法 使用函数修饰符来声明静态方法和类方法 静态方法 静态方法仅是类中的函数, 不需要绑定实例, 也就是说静态方法的定义不需要传入 self 参数. 静态方法不属于类的某一个实 ...
- Python 进阶_OOP 面向对象编程_self 的实例绑定
目录 目录 self 和绑定 调用非绑定的方法 self 和绑定 在 Python 中 self 变量是特殊的, 其用于在实例方法中引用该方法所绑定的实例, 换句话说就是 Python 在实例化对象时 ...
- Python进阶之面向对象编程
面向对象编程——Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 面向过程的程序设计把计算机 ...
- Python进阶之面向对象编程(二)
Python面向对象编程(二) .note-content {font-family: "Helvetica Neue",Arial,"Hiragino Sans GB& ...
- Python进阶之面向对象编程概述
Python面向对象编程(一) .note-content {font-family: "Helvetica Neue",Arial,"Hiragino Sans GB& ...
- python进阶(5):组合,继承
前两天我们认识了面向对象也对面向对象有了初步的认识今天我们先会说一点组合的进阶,今天重点是继承. 一.组合 组合只有一个例子因为组合只作为上一章的补充内容 #老师 课程 生日 class Course ...
随机推荐
- c#处理json格式类型的字符串
string channelGroup=[{"SpType":"1","BaseInfoId":["xxx"," ...
- docker 一小时快速入门之利用docker安装Redis
利用docker方式快捷安装redis 该方式默认下载的最新版本镜像,如需要下载指定版本在redis后面跟:版本号 docker pull redis 查看当前下载redis的镜像 docker im ...
- Map与JSON Conver
function strMapToObj(strMap) { let obj = Object.create(null); for (let [k,v] of strMap) { ...
- forEach究竟能不能改变数组的值
forEach究竟能不能改变数组的值 :https://blog.csdn.net/ZhengKehang/article/details/81281563 初学者每次提到Array对象的时候有些烦人 ...
- bfs(最短路径)
链接:https://ac.nowcoder.com/acm/contest/993/F来源:牛客网 Farmer John is leaving his house promptly at 6 AM ...
- 洛谷 P1972 [SDOI2009]HH的项链(树状数组,离线)
传送门 解题思路 因为是求区间的不同种类数,所以我们用树状数组(貌似并没有什么直接联系) (...表示到) 还是和原来一样,用s[i]来表示a[i-lowbit(i)]...a[i]的种类数. 因为有 ...
- 状压BFS
题意:1个机器人找几个垃圾,求出最短路径. 状压BFS,这道题不能用普通BFS二维vis标记数组去标记走过的路径,因为这题是可以往回走的,而且你也不能只记录垃圾的数量就可以了,因为它有可能重复走同一 ...
- How Many Answers Are Wrong (HDU - 3038)(带权并查集)
题目链接 并查集是用来对集合合并查询的一种数据结构,或者判断是不是一个集合,本题是给你一系列区间和,判断给出的区间中有几个是不合法的. 思考: 1.如何建立区间之间的联系 2.如何发现悖论 首先是如何 ...
- project euler-34
145是个奇怪的数字.由于1!+ 4! + 5! = 1 + 24 + 120 = 145. 请求出能表示成其每位数的阶乘的和的全部数的和. 请注意:由于1! = 1和2! = 2不是和,故它们并不包 ...
- 解决java.net.BindException: Address already in use(Bind failed)端口占用问题
问题描述: 解决办法: sudo lsof -i:20101ps -ef|grep 9905kill -9 9905ps -ef|grep 9905 ------------------------- ...