Python的继承是多继承机制,一个子类可以同时有多个直接父类;继承可以得到父类定义的方法,子类就可以复用父类的方法。

一、继承的语法

子类:实现继承的类。

父类(基类、超类):被继承的类。

子类继承父类是在定义子类时,将多个父类放在子类之后的圆括号内,如果定义类时,未指定这个类的直接父类,则默认继承object类,所以object类是所有类的父类(直接父类或者是间接父类)。

语法格式如下:

class SubClass (SuperClassl , SuperClass2 , ... ):
# 类定义部分

使用例子:

class Animal:
def dog(self):
print ('我有一只狗!他叫%s'%self.dog_name) class Child:
def son(self):
print ('我有两个儿子!') # 定义Myself类,继承Animal类和Child类
class Myself(Animal,Child):
def name(self):
print ('我叫小明!') M = Myself() # 创建Myself对象
M.dog_name = '大黄' # 通过Myself对象添加dog_name类变量
M.dog() # 调用Myself对象的dog()方法,打印 我有一只狗!他叫大黄
M.son() # 调用Myself对象的son()方法,打印 我有两个儿子!
M.name() # 调用Myself对象的name()方法,打印 我叫小明!

在上面的例子中,定义了Animal和Child两个父类,和一个继承Animal类和Child类的Myself类;创建Myself对象后,可以访问Myself对象的dog()方法和son()方法,说明Myself对象也具有dog()方法和son()方法,所以继承可以得到父类定义的方法,通过继承子类就可以复用父类的方法。

二、多继承

子类会通过继承得到所有父类的方法,那么如果多个父类中有相同的方法名,排在前面的父类同名方法会“遮蔽”排在后面的父类同名方法,例:

class Animal:
def name(self):
print ('我有一只狗,他叫大黄!') class Child:
def name(self):
print ('我有两个儿子!') class Myself(Animal,Child):
pass class Yourself(Child,Animal):
pass M = Myself()
M.name() # 打印 我有一只狗,他叫大黄!
Y = Yourself()
Y.name() # 打印 我有两个儿子!

上面例子中,Myself类继承了Animal类和Child类,当Myself子类对象调用name()方法时,因为Myself子类中没有定义name()方法,Python会先在Animal父类中搜寻name()方法,一旦搜寻到就会停止继续向下搜寻,所以运行的是Animal类中的name()方法;而Yourself子类中由于Child父类排在前面,所以运行的是Child类的name()方法。

三、重写

子类包含与父类同名的方法称为方法重写(方法覆盖),可以说是重写父类方法,也可以说子类覆盖父类方法。

例:

class Animal:
def name(self):
print ('我有一只狗,他叫大黄!') class Myself(Animal):
# 重写Animal类的name()方法
def name(self):
print ('我没有狗,我只有一只猫,他叫大白!') M = Myself() # 创建Myself对象
M.name() # 执行Myself对象的name()方法,打印 我没有狗,我只有一只猫,他叫大白!

上面的例子中,运行M.name()时执行的不再是Animal类的fly()方法,而是Myself类的name()方法。

在子类中重写方法之后,如果需要用到父类中被重写的实例方法,可以通过类名调用实例方法来调用父类被重写的方法。

例:

class Animal:
def name(self):
print ('我有一只狗,他叫大黄!') class Myself(Animal):
# 重写Animal类的name()方法
def name(self):
print ('我还有一只猫,他叫大白!') def pet(self):
Animal.name(self) # 调用被重写的父类方法name(),使用类名.实例名调用,需要手动传self
self.name() # 执行name()方法,会调用子类重写的name()方法 M = Myself() # 创建Myself对象
M.pet()
'''
打印
我有一只狗,他叫大黄!
我还有一只猫,他叫大白!
'''

  

四、super函数调用父类构造方法

如果子类有多个直接的父类,那么排在前面的构造方法会遮蔽排在后面的构造方法。

例:

class Animal:
def __init__(self,pet,name):
self.pet = pet
self.name = name def favourite_animal(self):
print ('我有一只%s,他叫%s!'%(self.pet,self.name)) class Fruit:
def __init__(self,kind):
self.kind = kind def favourite_fruit(self):
print ('我喜欢的水果是%s!'%self.kind) class Myself(Animal,Fruit):
pass M = Myself('狗','大黄') # 创建Myself对象
M.favourite_animal() # 调用Myself对象的favourite_animal()方法,打印 我有一只狗,他叫大黄!
M.favourite_fruit() # 调用Myself对象的favourite_fruit方法,由于未初始化Fruit对象的kind实例变量,报错 AttributeError: 'Myself' object has no attribute 'kind'

上面例子中,Myself子类继承了Animal父类和Fruit父类,由于Animal父类排在Fruit父类前面,所以Animal父类的构造函数遮蔽了Fruit父类的构造函数,运行M.favourite_animal()没有任何问题,当运行M.favourite_fruit()时,由于未初始化Fruit对象的kind实例变量,所以程序会报错。

解决以上问题,Myself应该重写父类的构造方法,子类的构造方法可以调用父类的构造方法,有以下两种方式:

1.使用未绑定方法,即: 父类名.__init__(self,参数1,参数2....)。

2.使用super()函数调用父类构造方法。

先查看一下super()函数的帮助信息,

>>> help(super)
Help on class super in module builtins: class super(object)
| super() -> same as super(__class__, <first argument>)
| super(type) -> unbound super object
| super(type, obj) -> bound super object; requires isinstance(obj, type)
| super(type, type2) -> bound super object; requires issubclass(type2, type)
| Typical use to call a cooperative superclass method:
| class C(B):
| def meth(self, arg):
| super().meth(arg)
| This works for class methods too:
| class C(B):
| @classmethod
| def cmeth(cls, arg):
| super().cmeth(arg)
...

通过帮助信息,可以看到,当调用父类的实例方法时,会自动绑定第一个参数self;当调用类方法时,会自动绑定第一个参数cls。

接下来修改上面的程序:

class Animal:
def __init__(self,pet,name):
self.pet = pet
self.name = name def favourite_animal(self):
print ('我有一只%s,他叫%s!'%(self.pet,self.name)) class Fruit:
def __init__(self,kind):
self.kind = kind def favourite_fruit(self):
print ('我喜欢的水果是%s!'%self.kind) class Myself(Animal,Fruit):
def __init__(self,pet,name,kind):
Fruit.__init__(self,kind) # 通过未绑定方法调用父类构造方法
super().__init__(pet,name) # 通过super()函数调用父类构造方法 M = Myself('狗','大黄','苹果')
M.favourite_animal() # 打印 我有一只狗,他叫大黄!
M.favourite_fruit() # 打印 我喜欢的水果是苹果!

  

Python——继承的更多相关文章

  1. python 继承中的super

    python继承中子类访问父类的方法(包括__init__)主要有两种方法,一种是调用父类的未绑定方法,另一种是使用super(仅仅对于新式类),看下面的两个例子: #coding:utf-8 cla ...

  2. python继承

    Python继承 继承实例: 父类和子类的关系: 继承树: 没有父类就继承object类,不要忘记调用super().__init__来初始化父类 代码: class Person(object): ...

  3. python继承——封装

    python继承--封装 1 为什么要封装 封装数据的主要原因是:保护隐私 封装方法的主要原因是:隔离复杂度 2 封装分为两个层面 第一个层面的封装(什么都不用做):创建类和对象会分别创建二者的名称空 ...

  4. python 继承与多重继承

    当然,如果不支持python继承,语言特性就不值得称为“类”.派生类定义的语法如下所示: <statement-1> . . . <statement-N> 名称 BaseCl ...

  5. python 继承和多态

    在OOP程序设计中,当我们定义一个class的时候,可以从某个现有的class继承,新的class称为子类(Subclass),而被继承的class称为基类.父类或超类(Base class.Supe ...

  6. python继承的实例

    class SchoolMember(object):#定义学校 member=0#默认成员为0个 amount=0#默认学费为0元 def __init__(self,name,age,sex):# ...

  7. python 继承基础

    class annamal: def chi(self): print(self.name + '吃') def he(self): print(self.name + '喝') class dog( ...

  8. Python继承扩展内置类

    继承最有趣的应用是给内置类添加功能,在之前的Contact类中,我们将联系人添加到所有联系人的列表里,如果想通过名字来搜索,那么就可以在Contact类添加一个方法用于搜索,但是这种方法实际上属于列表 ...

  9. Python继承、方法重写

    继承 在编写类时,并不是每次都要从空白开始.当要编写的类和另一个已经存在的类之间存在一定的继承关系时,就可以通过继承来达到代码重用的目的,提高开发效率. class one(): "&quo ...

  10. python继承和多态

    继承 目标 单继承 多继承 面向对象三大特性 封装 根据 职责 将 属性 和 方法 封装 到一个抽象的 类 中 继承 实现代码的重用,相同的代码不需要重复的编写 多态 不同的对象调用相同的方法,产生不 ...

随机推荐

  1. vue 错误提问示例

    > 表格显示数据,选中其中一条数据进行编辑,弹出层编辑,发现修改数据时,表格中的数据同步变化.编辑成功之后,还会返回原本没修改之前的数据. > 传对象给子组件,子组件的值污染父组件.

  2. SOA面向服务体系架构

    SOA概念 1.什么是SOA 面向服务的体系结构(Service-Oriented Architecture,SOA)是一个组件模型. 它将应用程序的不同功能单元(称为服务)通过这些服务之间定义良好的 ...

  3. [面试]快来测测你的C++水平

    在32位编译环境下进行测试. 以下代码运行结果是什么? #include <iostream> using namespace std; class D { public: static ...

  4. What is content-type and datatype in an AJAX request?

    https://api.jquery.com/jquery.ajax/ What is content-type and datatype in an AJAX request? contentTyp ...

  5. useContext 让父子组件传值更简单(五)

    有了useState和useEffect已经可以实现大部分的业务逻辑了,但是React Hooks中还是有很多好用的Hooks函数的,比如useContext和useReducer. 在用类声明组件时 ...

  6. JVM内存空间划分与作用

    虚拟机栈:Stack Fame 栈桢 程序计数器(Program Counter): 本地方法栈:主要用于处理本地方法 堆(Heap): JVM管理的最大一块内存空间 方法区(Method Area) ...

  7. 字节码(.class)文件的加载过程

    类加载 在Java代码中,类型的加载.连接与初始化过程都是在程序运行期间完成的. 类型可以是Class,Interface, 枚举等. Java虚拟机与程序的生命周期 在如下几种情况下,Java虚拟机 ...

  8. flutter Slider滑块组件

    滑块,允许用户通过滑动滑块来从一系列值中选择. import 'package:flutter/material.dart'; class SliderDemo extends StatefulWid ...

  9. java输出一个目录下的子目录

    java输出一个目录下的子目录 package com.vfsd.core; import java.io.File; public class ListDir { public static voi ...

  10. npm publish 一直报错 404

    在封装 zswui  react ui 组件库的时候,尝试了下 github的 packages 包设置,然后就给自己挖坑了. zswui   这是一个从零开始配置,实现组件开发测试的项目. 因为设置 ...