这篇文章我们来介绍一下 super,我相信大部分的人使用 super 都是使用这种方式;

# 就是我有一个 class 比如说是 Male,然后继承另外一个 class 比如是 Person,然后我在这个 Male 也就是它的子类的 init 函数里面用 super().__init__() 来调用它父类的初识化函数

from objprint import op

class Person:
def __init__(self, name):
self.name = name class Male(Person):
def __init__(self, name):
super().__init__(name)
self.gender = "male" m = Male('xiaoyang')
op(m) # 输出:
<Male 0x23669a19fa0
.gender = 'male',
.name = 'xiaoyang'
>

​ 在我们常用 super 的时候都通常会认为 super 是一个方法或者函数,但是实际上 super 是一个正儿八经的 class,它是一个内置内的名字,然后 super() 并不是调用了一个函数 ,super() 是建立 了一个 super 的对象

>>> type(super)
<class 'type'>

​ 尽管我们更常用的是 super() 括号里面什么都没有,但是 super 的完整版它里面应该是有两个参数,第一个参数是一个 type 也就是一个 class,第二个参数是一个 type 或者是一个 object,其中第二个参数决定了这一个函数绑定到那个 object 或者 class 上,同时第二个参数决定了使用那个 mro,而第一个参数决定了在 mro 链上 从哪个class 开始往后找,例如;

from objprint import op

class Person:
def __init__(self, name):
self.name = name class Male(Person):
def __init__(self, name):
# super().__init__(name)
super(Male, self).__init__(name)
self.gender = "male" m = Male('xiaoyang')
op(m) # 输出:
<Male 0x171f680afa0
.gender = 'male',
.name = 'xiaoyang'
> # 其实我们看刚才的 super().__init__(name) 它是等价于 super(Male, self).__init__(name) 的。

​ 那么这个super(Male, self)它是做了这样一个事情,首先它要从 self 这个 object 里面拿到 mro,然后他会找到第一个 argument,也就是 Male 在 mro 里所处的位置,那在当前的情况下 Male 就是最开始的那个(Male Person object)接下来他会从 Male 后面的那个 class 开始找,那它第一个找到的就是 Person,然后它就看Person 里面有没有__init__这个函数,然后发现有这个函数,然后它在把这个__init__绑定到 self 上,在这里可以理解为这个 Person 的__init__函数传进去的这个 self 就是 super 里面的这个 self,也就是说Person.__init__(self,name)这行代码等价于 super(Male, self).__init__(name)这行代码。

至于为什么不直接使用Person.__init__(self,name)是有几个原因:

  1. 在未来有可能会改变基类的名字,甚至会改变继承的方式,在这种情况下如果使用 super 的话就什么都不用管,应为他会自动追随这个 mro 找到正确的 class ,但是用这种命名的 class 的话就要全部修改,这样更容易引起错误
  2. 其实 super 是动态的,他会根据 self 的 mro 进行寻找,而 self 也就是传进来的这个 argument 它本身是动态的,也就是说同样一个函数里面,我用 super 在不改变这个函数的情况下我有可能会拿到不同的 class。

在来看这个示例:

from objprint import op

class Animal:
def __init__(self, age):
self.age = age class Person(Animal):
def __init__(self, age, name):
super().__init__(age)
self.name = name class Male(Person):
def __init__(self,age, name):
# super(Male, self).__init__(age, name)
super(Person, self).__init__(age, name)
self.gender = "male" m = Male(18, 'xiaoyang')
op(m)

​ 如果在 Male 中正常的使用它 super(Male, self).__init__(age, name) ,那么它就会正常的初始化所有的东西,它会访问这个 Person 的 __init__ ,然后 Person 的__init__会访问 Animal 的__init__,最后就完成了这个 Male。

​ 那如果把它改成super(Person, self).__init__(age, name) ,那么就会报错,因为当我们使用 super(Person, self)的时候,self 的 mro 链是 Male Person Animal 然后是 object,那第一个参数它由于是 Person,所以他会从 Person 后面的那个 class 也就是 Animal 开始找,那 Animal 是有__init__函数的,但是 Animal 的 __init__ 只有一个参数 age,所以当我们传入 age name 的时候那就错了,这时候就只需要将它改成只传进去一个 age 如: super(Person, self).__init__(age) 就可以了,同时也跳过了 Person。

总结 super 的两个参数也就是第一个 type 和第二个 type 或者 object 分别决定了什么:

第一个只决定了在 mro 这个链上从哪里开始找

第二个是决定使用这个函数的对象和 mro

super 并不是只能在 class 里面使用的,它可以在任何一个地方使用,我只要给定 第二个参数 object 或者 class ,在给定第一个参数从哪里开始找,我就能使用它的函数,例如:

# 那这里的话就是从 m 这个 object 的 mro 上寻找 Male 后面开始的 __init__ 函数,这样实际上就找到了 Person 的 __init__ 函数,然后再用 Person 的 __init__ 函数对 m 这个 object 做初始化

from objprint import op

class Animal:
def __init__(self, age):
self.age = age class Person(Animal):
def __init__(self, age, name):
super().__init__(age)
self.name = name class Male(Person):
def __init__(self,age, name):
super(Person, self).__init__(age)
self.gender = "male" m = Male(18, 'xiaoyang')
op(m) print("----------------------") super(Male, m).__init__(20, "xiaoyang")
op(m) # 输出:
<Male 0x18412d13f70
.age = 18,
.gender = 'male'
>
----------------------
<Male 0x18412d13f70
.age = 20,
.gender = 'male',
.name = 'xiaoyang'
>

Python中的Super详解的更多相关文章

  1. 第7.19节 Python中的抽象类详解:abstractmethod、abc与真实子类

    第7.19节 Python中的抽象类详解:abstractmethod.abc与真实子类 一.    引言 前面相关的章节已经介绍过,Python中定义某种类型是以实现了该类型对应的协议为标准的,而不 ...

  2. python中threading模块详解(一)

    python中threading模块详解(一) 来源 http://blog.chinaunix.net/uid-27571599-id-3484048.html threading提供了一个比thr ...

  3. Python中time模块详解

    Python中time模块详解 在平常的代码中,我们常常需要与时间打交道.在Python中,与时间处理有关的模块就包括:time,datetime以及calendar.这篇文章,主要讲解time模块. ...

  4. python中常用模块详解二

    log模块的讲解 Python 使用logging模块记录日志涉及四个主要类,使用官方文档中的概括最为合适: logger提供了应用程序可以直接使用的接口API: handler将(logger创建的 ...

  5. 76.Python中F表达式详解

    F表达式是用来优化ORM操作数据库的. 举个例子:我们做口罩的公司要将所有员工的薪水增加2000元,如果按照正常的流程,应该是先从数据库中提取所有的员工的工资到Python内存中,然后使用Python ...

  6. python 中的unicode详解

    通过例子来看问题是比较容易懂的. 首先来看,下面这个是我新建的一个txt文件,名字叫做ivan_utf8.txt,然后里面随便编辑了一些东西. 然后来用控制台打开这个文件,同样也是截图: 这里就是简单 ...

  7. 4、Python中的类详解(0601)

    <大话数据结构>的作者程杰在博客园也有博客,网址是:http://cj723.cnblogs.com/ 面向对象编程(OOP) 1.程序 = 指令 + 数据 代码可以选择以指令为核心或以数 ...

  8. python 中model.py详解

    model详解 Django中遵循 Code Frist 的原则,即:根据代码中定义的类来自动生成数据库表. 创建表 基本结构 from django.db import models # Creat ...

  9. python 中的map 详解

    python中的map函数应用于每一个可迭代的项,返回的是一个结果list.如果有其他的可迭代参数传进来,map函数则会把每一个参数都以相应的处理函数进行迭代处理.map()函数接收两个参数,一个是函 ...

随机推荐

  1. ssh端口转发学习笔记

    ssh端口转发学习笔记 ssh命令参数介绍 -C 压缩数据传输 -f 将 ssh 转到后台运行,即认证之后,ssh 自动以后台运行.不在输出信息 -n 将 stdio 重定向到 /dev/null,与 ...

  2. 【Java分享客栈】Java程序员为争一口气熬夜硬刚CSS实现掘金首页

    前言 如果我做不了最厉害的Java工程师,那我就做Java工程师中最厉害的前端工程师. 前段时间,我默默给自己又喂了这碗心灵鸡汤-- 我不是很厉害的Java工程师,哪怕我已经工作八年,我依然觉得自己和 ...

  3. jvm-learning-概述

    JVM整体结构: java代码的执行流程 JVM的架构模型 public class StackStruTest { public static void main(String[] args) { ...

  4. Mybatis框架基础入门(一)--简介及优势

    一.什么是Mybatis 这里借用官网的一句话介绍什么是mybatis:MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC ...

  5. Oracle入门基础(九)一一创建表和管理表

    练习:查询每一年入职人数及总人数 SQL> select count(*) Total, 2 sum(decode(to_char(hiredate,'yyyy'),'1980',1,0)) & ...

  6. java-設計模式-抽象工場模式

    抽象工廠模式AbstractFactory 一种创建型设计模式, 它能创建一系列相关的对象, 而无需指定其具体类. 工廠方法模式中考虑的是一类产品的生产,如畜牧场只养动物.电视机厂只生产电视机,同种类 ...

  7. vue集成CKEditor构建框架的使用,遇到富文本框不出现工具栏等操作

    官方关于Vue集成CKEditor富文本框的文档:https://ckeditor.com/docs/ckeditor5/latest/builds/guides/integration/framew ...

  8. Jpa 在CriteriaBuilder中添加where条件NotIn子查询

    final CriteriaBuilder cb = entityManager.getCriteriaBuilder(); final CriteriaQuery<Person> cq ...

  9. STM32试题及答案

    一.选择题 1.Cortex-M处理器采用的架构是(  D  ) (A)v4T               (B)v5TE              (C)v6                (D)v ...

  10. leetcode_两数相加

    给你两个 非空 的链表,表示两个非负的整数.它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字. 请你将两个数相加,并以相同形式返回一个表示和的链表. 你可以假设除了数字 0 ...