一、继承

  原则:大部分使用继承的场合都可以用组合取代或者简化,而多重继承则需要不惜一切的避免。

  1. 什么是继承

  继承:用于指明一个类的大部分或者全部功能都是从一个父类获得的。此时,父类的实例的所有动作可能都会工作在子类的实例上,这样可以把通用的功能放在父类里边,然后给子类设定一些特别的功能。

  父类和子类的三种交互方式:

  ① 子类上的动作完全等同于父类上的动作

  ② 子类上的动作完全覆盖了父类上的动作

  ③ 子类上的动作部分替换了父类上的动作

  2. 隐式继承

  如果在父类中定义了一个函数,而在子类中没有定义,则会发生隐式的继承。也就是说,子类的实例可以调用仅在父类中定义过而在子类中没有定义过的函数。

 class Parent(object):

     def implicit(self):
print("PARENT IMPLICIT()") class Child(Parent):
pass dad = Parent()
son = Child() dad.implicit()
son.implicit()

  输出为

PARENT IMPLICIT()
PARENT IMPLICIT()

  3. 显式覆盖

  如果想要在子类里的函数有不同的功能,则必须使用显式覆盖,这在子类中定义一个同名函数就可以了,如

 class Parent(object):

     def override(self):
print("PARENT OVERRIDE()") class Child(Parent): def override(self):
print("CHILD OVERRIDE()") dad = Parent()
son = Child() dad.override()
son.override()

  输出为

PARENT OVERRIDE()
CHILD OVERRIDE()

  4. 在运行前或运行后替换

  替换是覆盖的一种特例。它允许我们在父类定义的内容运行之前或之后再去修改子类的行为。

  其方法为,首先像上例一样覆盖父类的函数,然后用super()调用该函数在父类里的版本。

 class Parent(object):

     def altered(self):
print("PARENT altered()") class Child(Parent): def altered(self):
print("CHILD, BEFORE PARENT altered()") # 显式覆盖
super(Child,self).altered() # 用super(Child,self)调用父类的同名函数
print("CHILD, AFTER PARENT altered()") # 仍为显式覆盖的部分 dad = Parent()
son = Child() dad.altered()
son.altered()

  输出为

PARENT altered()
CHILD, BEFORE PARENT altered()
PARENT altered()
CHILD, AFTER PARENT altered()

  可见,一方面,我们可以显式覆盖,重写子类同名函数的内容和功能;另一方面,可以用super(Child,self)函数在中间调用父类的同名函数,而它并不影响前后的覆盖内容。

  因此,我们可以在子类函数运行前或运行后替换,这取决于super()的位置。

  5. 三种方式的组合使用

 class Parent(object):

     def implict(self):
print("PARENT implict()") def override(self):
print("PARENT override()") def altered(self):
print("PARENT altered()") class Child(Parent): def override(self):
print("CHILD override()") def altered(self):
print("CHILD, BEFORE PARENT altered()")
super(Child, self).altered()
print("CHILD, AFTER PARENT altered()") dad = Parent()
son = Child() dad.implict()
son.implict() dad.override()
son.override() dad.altered()
son.altered()

  输出为

PARENT implict()
PARENT implict()
PARENT override()
CHILD override()
PARENT altered()
CHILD, BEFORE PARENT altered()
PARENT altered()
CHILD, AFTER PARENT altered()

  6. super()函数

  ① 多重继承

  指定义的类继承了一个或多个类,例如

class SuperFun(Child, BadStuff):
pass

  此时,一旦在SuperFun的实例上调用任何隐式动作,Python就必须返回到Child和BadStuff的类层次结构中去查找可能的函数,而且必须用固定的顺序去查找。

  为此,Python使用了“Method Resolution Oder(方法解析顺序),MRO”+C3算法,并提供super()函数,用于实现这些功能。这样,就可以方便的使用super()来查询各父类的函数了。

  ② super()与__init__的搭配使用

  super()最常见的用法是,在父类的__init__函数中使用。它使得我们可以先在子类中实现一些功能,然后再让父类初始化,例如

 class Child(Parent):

     def __init__(self, stuff):
self.stuff = stuff
super(Child, self).__init__()

  二、组合

  组合能够实现和继承几乎完全相同的功能,但它采用的方法并非是隐式继承,而是通过调用模块里的函数来实现。

 class Other(object):

     def implicit(self):
print("OTHER implicit()") def override(self):
print("OTHER override()") def altered(self):
print("OTHER altered()") class Child(object): def __init__(self):
self.other = Other() def implicit(self):
self.other.implicit() def override(self):
print("CHILD override()") def altered(self):
print("CHILD, BEFORE OTHER altered()")
self.other.altered()
print("CHILD, AFTER OTHER altered()") son = Child() son.implicit()
son.override()
son.altered()

  输出为

OTHER implicit()
CHILD override()
CHILD, BEFORE OTHER altered()
OTHER altered()
CHILD, AFTER OTHER altered()

  可见,这里并不是为Child指定了一个父类,而是将父类的功能打包为一个模块(实质也是一个类),然后直接调用模块里面的函数,这也能够完成和继承相同的功能。

  三、继承和组合的应用场合

  继承和组合事实上都是为了解决代码的复用问题。前者通过创建一种让我们在子类里隐含父类的功能的机制来解决这个问题,后者通过模块/类中的函数调用达到了相同的目的。关于二者的选择,大体上有3个指导原则:

  1. 避免多重继承。多重继承复杂且不可靠,如果要使用,需要钻研一下类层次结构及其来龙去脉。

  2. 如果有一些代码会在不同位置和场合应用到,就用组合将它们做成模块。

  3. 只有在代码的可复用部分之间有清楚的关联,可以通过一个单独的共性连接起来的时候,才使用继承。

【巩固练习】代码风格参考 PEP 8 -- Style Guide for Python Code

【注意】在有的语言如JavaScript中,对象其实是类的副本,这样的语言叫原型(prototype)语言,这种语言中类和对象除了用法以外没多少不同。但在Python中,类则比较像是用来创建对象的模板。

【Python基础】lpthw - Exercise 44 继承与组合的更多相关文章

  1. Python 入门基础20 --面向对象_继承、组合

    今日内容 组合:自定义类的对象作为类的属性 继承:父类与子类.多继承 1.组合 将自定义类的对象作为类的属性 class Teacher: def __init__(self, name): self ...

  2. python基础学习笔记—— 多继承

    本节主要内容: 1.python多继承 2.python经典类的MRO 3.python新式类的MRO.C3算法 4.super是什么鬼? 一.python多继承 在前⾯的学习过程中. 我们已经知道了 ...

  3. python基础学习笔记——单继承

    1.为什么要有类的继承性?(继承性的好处)继承性的好处:①减少了代码的冗余,提供了代码的复用性②提高了程序的扩展性 ③(类与类之间产生了联系)为多态的使用提供了前提2.类继承性的格式:单继承和多继承# ...

  4. python基础学习之类的继承、魔法方法

    什么是继承 即类A可以使用类B的方法,即B是A的父类,A是B的子类,AB之间是继承关系 class Father():  # 父类 def __init__(self,name,age): self. ...

  5. python基础(17)继承类和面向对象初识

    1.继承类 class Lm: money = 1000000 house = 5 def driver(self): print('会开车') class Mcb(Lm): def about_me ...

  6. python基础——继承与派生、组合

    python基础--继承与派生 1 什么是继承: 继承是一种创建新的类的方式,在python中,新建的类可以继承自一个或者多个父类,原始类成为基类或超累,新建的类成为派生类或子类 1.1 继承分为:单 ...

  7. python基础----继承与派生、组合、接口与归一化设计、抽象类、子类中调用父类方法

    一.什么是继承                                                                          继承是一种创建新的类的方式,在pyth ...

  8. python基础之类的继承与派生、组合、接口与归一化设计、抽象类、子类中调用父类方法

    一.什么是继承 继承是一种创建新的类的方式,新建的类可以继承自一个或者多个父类,原始类称为基类或超类,新建的类称为派生类或子类. 派生:子类继承了父类的属性,然后衍生出自己新的属性,如果子类衍生出的新 ...

  9. python基础之继承派生、组合、接口和抽象类

    类的继承与派生 经典类和新式类 在python3中,所有类默认继承object,但凡是继承了object类的子类,以及该子类的子类,都称为新式类(在python3中所有的类都是新式类) 没有继承obj ...

随机推荐

  1. thymeleaf时间格式化

    Thymeleaf模板时间格式表达式     ${#dates.format(date, 'dd/MMM/yyyy HH:mm')} 例如: <input name="enroDate ...

  2. shiro-core包引用的版本问题

    在做shiro学习时,遇到这样的问题: Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/ ...

  3. Gradle part1 HelloWorld

    (https://spring.io/guides/gs/gradle/#scratch) ----gradle helloworld----- 1.下载后安装 Unzip the file to y ...

  4. iOS 中的屏幕旋转shouldAutorotate和supportedInterfaceOrientations的先后关系

    这2个UIViewController的属性,都和旋转相关, 当设备发生旋转时,首先会查看根controller的shouldAutorotate是否允许旋转,如果允许,再通过 supportedIn ...

  5. 小程序生成海报图片(或者原有的)并下载,&&相册授权&&按钮拉起二次授权

    这是自己做小程序生成推广海报,并保存到本地相册的方法,向后台发起请求,返回一个海报图片,下载保存到相册, 如果只是单纯的下载图片代码43行-63行就足够了 如果想直接保存到相册,则不要做downFil ...

  6. python下的异常处理

    一.什么是异常 程序运行过程中错误发生的信号.(如果运行时产生的异常,程序不处理就会被抛出,随之会造成程序终止) 二.异常的种类 首先,异常主要分为语法错误.逻辑错误两种类型 语法错误会在程序还没有执 ...

  7. C语言fread/fwrite填坑记

    坑的描述 用fwrite把数据写入文件,再用fread读取,发现后半部分的数据可能是错的. 原因:原本要写入文件的数据中,有0x0A,如果用的是文本模式打开的文件流,在windows下0x0A会被转换 ...

  8. 【原创】sqlite ef6 踩坑

    调试的时候配置写如下,这样写是没有问题的但是在实际环境中有问题,因为EF路径找不到.会提示错误:The underlying provider failed on open <connectio ...

  9. [PKUSC2018]星际穿越

    [PKUSC2018]星际穿越 题目大意: 有一排编号为\(1\sim n\)的\(n(n\le3\times10^5)\)个点,第\(i(i\ge 2)\)个点与\([l_i,i-1]\)之间所有点 ...

  10. centos中安装基础环境

    进入到相关目录cd /usr/bin安装python3yum install zlib-devel bzip2-devel openssl-devel ncurses-devel sqlite-dev ...