Python的程序结构[2] -> 类/Class[1] -> 基类与继承
基类与继承 / Base Class and Inheritance Class
面向对象的特性使得 Python 中不可避免地需要使用到类和类的继承,类的继承可以使得代码很好的被重用。下面以一些代码示例说明类的继承如何使用。
继承一个基类
首先,定义一个基类 Animal,在初始化中设定一个基本属性以及物种信息,并设置其具有 eat 的能力(self.eat 为 True)。此处还重载了魔术方法 __getattr__,当搜索的属性不存在时返回 False(即不具备该能力),最后定义一个 show 函数来显示当前 species 信息。
# ------- Basic inheritance --------
class Animal:
def __init__(self):
self.base = "Creature"
self.species = "Animal"
self.eat = True def __getattr__(self, item):
return False def show(self):
print("This is %s." % self.species)
接着定义一个 Bird 类,继承自 Animal ,在 Bird 的初始化函数中,首先对基类进行初始化,然后设置 species 属性,以及 fly 为True(代表具有 fly 的能力),并对 Bird 定义了 show 函数。同样定义 Fish 类,设置 species 信息以及 swim 为 True,且不定义 show 函数。
# Note: Do not mix super and Object.__init__
class Bird(Animal):
def __init__(self):
Animal.__init__(self)
self.species = "Bird"
self.fly = True def show(self):
print("This is %s from %s." % (self.species, self.base)) class Fish(Animal):
def __init__(self):
Animal.__init__(self)
self.species = "Fish"
self.swim = True
最后,对两个类分别进行实例化,并调用各自的 show 函数,
b = Bird()
f = Fish()
b.show()
f.show()
输出的结果如下,
This is Bird from Creature.
This is Fish.
从最终的输出结果可以看到,
对于 Bird 类,并没有对 base 属性进行定义,但却拥有 base 属性,即这一属性从基类 Animal 中继承得到,
而对于 Fish 类,并未定义 show 方法,但是却在实例中可以使用 show 方法。同样这一方法也是继承自基类 Animal。
这两个简单的示例中通过继承从而避免了重复的代码和定义。
多继承
前面的例子中使用的基类是唯一的,当需要从多个基类中继承时,则会涉及到多继承问题。下面的代码给出一个多继承问题的示例。
首先导入之前定义的两个类 Bird 和 Fish 作为基类,然后基于这两个类派生出子类 Duck 和 Goose,其中 Goose 中重载了初始化函数,利用 super 初始化 Goose 的基类。此时,我们希望 Duck 和 Goose 能够继承 Bird 和 Fish 的两个特性,fly 和 swim。
Note: 此处为示例代码,切勿混用 super 和类的初始化函数。
from inheritance_base import Bird, Fish
# ------- Multi inheritance --------
class Duck(Bird, Fish): pass class Goose(Bird, Fish):
def __init__(self):
super(Goose, self).__init__() d = Duck()
g = Goose()
print("I am %s, I can fly: %s" % (d.species, d.fly))
print("I am %s, I can swim: %s" % (d.species, d.swim))
print("I am %s, I can fly: %s" % (g.species, g.fly))
print("I am %s, I can swim: %s" % (g.species, g.swim))
最终的运行结果如下,
I am Bird, I can fly: True
I am Bird, I can swim: False
I am Bird, I can fly: True
I am Bird, I can swim: False
从运行的结果中看到,最后的输出却不是我们所期望的,无论是 Duck 的实例还是 Goose 的实例都不具备 swim 的属性,也就是说,Fish 的初始化并未被执行。
其原因在于,Duck 中并未定义初始化函数,因此会在父类中搜索,从而调用到 Bird 的初始化函数,所以最终显示 species 属性为 Bird,而 swim 为 False。而在 Goose 中,使用 super 也未能成功继承所有属性,这是由于 super 会依照 MRO 顺序进行搜索,使用搜索到的第一个类的对应方法。
为实现所需要的功能,在这里重新定义一个 Duck 类,重载初始化函数,在初始化函数中分别调用两个基类的初始化函数。
class Duck(Bird, Fish):
def __init__(self):
Bird.__init__(self)
Fish.__init__(self) d = Duck()
print("I am %s, I can fly: %s" % (d.species, d.fly))
print("I am %s, I can swim: %s" % (d.species, d.swim))
最终输出结果可以看到,species 属性被覆盖了最终显示为 Fish,而 Goose 也具备了 fly 和 swim 两个属性。
I am Fish, I can fly: True
I am Fish, I can swim: True
但这种继承依旧会引起一些问题。
菱形 / 钻石继承
前面的多继承使用直接调用父类初始化函数进行,表面上能够满足多继承的需求,但是仍然存在一些问题,下面以一个菱形/钻石继承来说明这一问题是如何存在的。
首先是定义 A、B、C、D 四个类,这四个类大致为菱形继承的形式。
"""
A
/\
/ \
/ \
B C
\ /
\ /
\/
D
"""
然后以 no-super 和 super 两种方式来实现这一继承,
在 no super 中不使用 super,直接使用调用父类初始化函数的方法来完成初始化工作,具体方式如下,
# ------- no super ----------
class A(object):
def __init__(self):
object.__init__(self)
print("This is A init.") class B(A):
def __init__(self):
A.__init__(self)
print("This is B init.") class C(A):
def __init__(self):
A.__init__(self)
print("This is C init.") class D(B, C):
def __init__(self):
B.__init__(self)
C.__init__(self)
print("This is D init.") d = D()
输出结果如下,从输出中可以看出,这种方式存在一个问题,那就是对 B 和 C 的基类 A 进行了多次的初始化。这可能会造成一些不期望的结果。
This is A init.
This is B init.
This is A init.
This is C init.
This is D init.
为了避免这种现象,可以使用 super 来完成继承初始化。
# ------- super ------------
class A(object):
def __init__(self):
super(A, self).__init__()
print("This is A init.") class B(A):
def __init__(self):
super(B, self).__init__()
print("This is B init.") class C(A):
def __init__(self):
super(C, self).__init__()
print("This is C init.") class D(B, C):
def __init__(self):
super(D, self).__init__()
print("This is D init.") d = D()
使用super的初始化结果显示如下,可以看到,此时能够实现对 A 的初始化只进行一次。这是由于 super 中实现了 MRO 的搜索算法。
This is A init.
This is C init.
This is B init.
This is D init.
相关阅读
2. super
3. MRO 顺序
Python的程序结构[2] -> 类/Class[1] -> 基类与继承的更多相关文章
- Python+Selenium框架设计之框架内封装基类和实现POM
		
原文地址https://blog.csdn.net/u011541946/article/details/70269965 作者:Anthony_tester 来源:CSDN 博客地址https ...
 - 请高手解释这个C#程序,其中ServiceBase是windows服务基类,SmsService是
		
请高手解释这个C#程序,其中ServiceBase是windows服务基类,SmsService是 ServiceBase的子类. static void Main() { ServiceBase[] ...
 - 派生类地址比基类地址少4(子类与基类指针强行转换的时候,值居然会发生变化,不知道Delphi BCB是不是也这样) good
		
大家对虚表并不陌生,都知道每个含有虚函数的类对象都有1个虚指针,但是在现实使用中,却总是因为这而调试半天,才发现原来是虚指针惹的祸.我这几天在调试代码时候也中招了,我的问题是这样的,如下图,CTree ...
 - 【Android进阶】为什么要创建Activity基类以及Activity基类中一般有哪些方法
		
现在也算是刚刚基本完成了自己的第一个商业项目,在开发的过程中,参考了不少人的代码风格,然而随着工作经验的积累,终于开始慢慢的了解到抽象思想在面向对象编程中的重要性,这一篇简单的介绍一下我的一点收获. ...
 - 派生类地址比基类地址少4(CDerived对象的起始地址存放的是虚表指针vptr,也就是子类的第一项内容。接下来的是基类的成员变量,接下来再是自身的成员变量)
		
大家对虚表并不陌生,都知道每个含有虚函数的类对象都有1个虚指针,但是在现实使用中,却总是因为这而调试半天,才发现原来是虚指针惹的祸.我这几天在调试代码时候也中招了,我的问题是这样的,如下图,CTree ...
 - 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)
		
[源码下载] 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成 ...
 - C++ - 派生类访问模板基类(templatized base class)命名
		
派生类访问模板基类(templatized base class)命名 本文地址: http://blog.csdn.net/caroline_wendy/article/details/239936 ...
 - laravel5.8笔记五:基类控制器和基类模型
		
建立基类的目的就是为了方便继承.比如:Admin模块访问,是否登陆.检测登陆可以写到基类里面 控制器基类 原始基类:app\Http\Controllers\Controller.php,我们下面要做 ...
 - C++ - 派生类强制转换为基类
		
派生类强制转换为基类 本文地址: http://blog.csdn.net/caroline_wendy/article/details/24268821 在多态的使用时, 派生类的指针或引用能够转换 ...
 - C#在派生类中调用基类成员
		
一.在派生类中调用基类成员 在C#的派生类中,我们可以使用base关键字调用基类中的公有或者受保护成员.这些成员只能是构造函数.实例方法或者实例属性. base关键字调用基类成员的语法格式如下: ba ...
 
随机推荐
- 打开Vim/Vi代码高亮
			
由于新装Vim/Vi 默认是没有打开代码高亮配置的,就看到有朋友一次次到网上去找各种配置.其实Vim默认带来配置文件的样本的,只需拷贝过来就可使用. 在用户根目录(~)中新建vim的配置文件 .vim ...
 - APPIUM-----自动发现兼容的Chromedrivers
			
使用Appium Desired Capabilities:chromedriverExecutableDir chromeDriver所有版本下载路径:https://chromedriver.st ...
 - Jmeter微信小程序接口测试
			
最近公司新项目组开发一款微信小程序电商平台,为了更好保证产品质量,因此提出了需要进行接口测试. 从接口本身来讲,对其测试与其他项目应该是一样的.所以不难理解,我们要对小程序的接口测试需要准备的 材料有 ...
 - shell监控脚本
			
序言: 前几天一好友问我服务器监控怎么做?你们公司的监控是怎么做的?有什么开源的监控软件推荐?常见的开源的监控软件当然首先推荐ZABBIX,分布式够强大,而且很多公司都在用,我问他具体什么需求,能监控 ...
 - (原)  Unreal搬山-引言(图多慎)
			
@author:白袍小道 扯淡:(图多) 何为搬山,这里借了剑来少年郎一句.(若有同道中人,甚是开心,开心的很) 江湖岂能没前辈) (江湖很大,足够你浪) (刺客信条 \荒野 \神秘海域 \死亡空间 ...
 - c语言版贪吃蛇小游戏
			
编译环境:windows 7 64位 编译工具:codeblocks 13.12 备注:未使用graphics.h 声明:个人原创,未经允许,禁止转载!!! 数据结构:双向链表 1.程序未使用grap ...
 - ZOJ 3606 Lazy Salesgirl ( 线段树 + 思路 )
			
卖切糕的小女孩 http://www.cnblogs.com/wuyiqi/archive/2012/04/28/2474672.html #include <cstdio> #inclu ...
 - 团队项目-任务分解[Alpha0]
			
团队项目-任务分解[Alpha0] 标签(空格分隔): 团队博客 适用范围: 本文档 适用对象 团队全体成员 适用时间 alpha阶段第一周计划 10.24-10.28 适用内容 目标.分工.时长估计 ...
 - Nginx和Squid配合搭建的Web服务器前端系统
			
这个架构是目前我个人觉得比较稳妥并且最方便的架构,易于多数人接受: 前端的lvs和squid,按照安装方法,把epoll打开,配置文件照搬,基本上问题不多. 这个架构和app_squid架构的区别,也 ...
 - RabbitMQ vhost 配置
			
RabbitMQ vhost 配置 rabbitmqctl set_vhost_limits是用来定义虚拟主机限制的命令 配置最大连接限制 要限制vhost vhost_name中并发客户端连接的 总 ...