我们最常见的,可以说几乎唯一能见到的使用super的形式是:

  1. class SubClass(BaseClass):
  2. def method(self):
  3. super(SubClass, self).method()
  4. #do some thing here...

其中,最常见的method()是__init__()

以前做过Java,刚刚接触Python的继承,的确感觉有点奇怪,有时候还能搞混。。。当然这还是因为Java是单继承,只能用接口实现类似多继承的方式,而Python就直接支持多继承。

当我第一次见到这个super()的时候,我想说,这是啥啊?为什么Python的super看起来这么奇怪呢?
super直接指向父类的实例不就得了么?为什么非得搞两个参数?
而且第一个参数还是自己的类名,而不是父类的类名!
既然是自己的类名你还费劲写一遍干嘛,干脆省略不得了?
第二个参数为啥是self?既然是self你省略不得了?
还有,super()返回的是个啥玩意啊,为什么可以.method() ? 难道他返回了一个BaseClass的instance么?
 
今天就来解答这一连串的问题。
 

1)super直接指向父类的实例不就得了么?

首先是不能指向,因为子类实例里面根本就没有一个父类实例,这跟java是不同的。详细看这里
其次是即便像我们想象的,子类实例里有一个叫super的指针,指向父类的实例(这个父类实例在子类实例生成的时候自动生成了,是子类实例的一个‘成员’)。
但是由于多继承的存在,还是会有问题。这也是跟java不同,java不支持多继承,java用接口的方式实现多继承。
举个例子:
当只存在单继承的时候,一切很美好,就像java一样:

你只需要在重写方法的时候,用super去调用父类的方法,然后再做自己的事就可以了。
 
但是在多继承的情况下,一切都不美好了:
 

当艺人展示自己能力的时候,他说,父亲啊,你先展示自己的能力,然后我来!
但是父亲是谁? 他有两个父亲! 总不能写两遍 super.展示自己 吧?
那么我们自然想到,给super搞一个参数不得了,指出super的是哪个父类。
比如 super(歌手).展示自己(),super(演员).展示自己(),好像一切又可以美好了:
 

先不说这种写法多么的丑陋,重复了那么多无意义的、类似的行,单说这种方法根本就行不通。

我们看看如果某个艺人的实例调用了.展示能力() ,将会输出什么:
  1. 我是人类,我能走路!
  2. 我是歌手,我能唱歌!
  3. 我是人类,我能走路!
  4. 我是演员,我能演戏!
  5. 我是艺人,我能代言广告!
有必要告诉大家两遍你是个人类么?这要是一个又身兼了主持人的艺人,岂不是得大声宣布三次自己是个人类??
这显然是不好的。
 
而且,这只是其一,假设多说了几遍自己是人类你也能忍,那么还有一种情况,更糟。
你比如说,艺人类没有重写 展现自己() 这个方法,当你用艺人的实例去调用 展现自己()的时候,将会发生啥?
不重写的方法自动调用父类的,但是有两个父类呀!调用哪个的呢?任意调用一个(比如按从左到右的顺序)?
那么体现不出他另一个父类的特点呀!比如调用歌手类的,那只能体现出这个艺人既是个歌手,又是个人类,根本体现不出他是个演员!
两个都调用?那万一其中一个父类没有这个方法呢?而且先调用哪个后调用哪个?按照左右顺序么?
总之这样很乱,编译器表示压力很大。
 

 
 
所以,python的super()用了两个参数。
先写在这里:
 
super(SubClass, self).method() 的意思是,根据self去找SubClass的‘父亲’,然后调用这个‘父亲’的method()
 
最最重点的地方来了:对于一个类来说,它的‘父类’到底是谁,取决于站在谁的角度去看。
 
 

‘父亲’之所以加了引号,是因为‘父亲’并不一定是上下级的继承关系,很可能只是兄弟。比如上图中歌手的‘父亲’是‘演员’,但实际上他们是兄弟关系。
所以以后用‘下一个类’来代替这个带引号的‘父亲’。
 
总之,站在不同的实例的角度,看到的‘下一个类’是不同的。
 
这就是super()第二个参数————self 的意义。
如果是一个歌手实例调用的,那么self=歌手实例,super()将站在一个歌手的角度,去找歌手类的‘下一个类’,则会找到人类;

如果是一个艺人实例调用的,那么self=艺人实例,super()将站在一个艺人的角度,去找歌手类的‘下一个类’,则会找到演员。

 
 代码如下:
  1. class People(object):
  2. def show_my_power(self):
  3. print(u"I am a people, I can walk !")
  4.  
  5. class Singer(People):
  6. def show_my_power(self):
  7. super(Singer, self).show_my_power()
  8. print(u"I am a singer, I can sing !")
  9.  
  10. class Actor(People):
  11. def show_my_power(self):
  12. super(Actor, self).show_my_power()
  13. print(u"I am an actor, I can act !")
  14.  
  15. class Artist(Singer, Actor):
  16. pass
  17.  
  18. if __name__ == "__main__":
  19. a = Artist()
  20. a.show_my_power()

运行结果:

  1. I am a people, I can walk !
  2. I am an actor, I can act !
  3. I am a singer, I can sing !
这里需要注意:当一个类从多个类继承的时候,按照从左到右的顺序。
也就是说,站在艺人的角度,首先看到艺人类的‘下一个类’是歌手类,然后歌手类的‘下一个类’是演员类。
这是因为歌手写在了左边:Artist(Singer, Actor)

看一下上面代码的执行过程吧:

由于 super() 绝大多数时候,都在咱们开篇说的这种情况下使用,即在类的定义语句块内部写:super(本类名,self)
所以 python3 做了个简化,如果你在类定义的语句块内写一个不带参数的super(),则相当于写了 super(本类名,self)
 

因为 super() 不光可以用在类的定义内部,所以,这种方便仅在类定义体内部有效。

参考

 

Python面试题之Python的Super方法的更多相关文章

  1. 千万不要错过这几道Python面试题,Python面试题No16

    第1题: python下多线程的限制以及多进程中传递参数的方式? python多线程有个全局解释器锁(global interpreter lock),简称GIL,这个GIL并不是python的特性, ...

  2. Python面试题之Python和Java中Super方法的区别

    python 的 super 是一个函数,需要两个参数,第一个参数是类,第二个参数是实例,返回值是一个类对象. 其意义是:站在参数2这个实例的角度看去, 参数1这个类的‘父亲’是谁,把‘父亲’返回. ...

  3. Python面试题之Python面试题汇总

    在这篇文章中: Python基础篇 1:为什么学习Python 2:通过什么途径学习Python 3:谈谈对Python和其他语言的区别 Python的优势: 4:简述解释型和编译型编程语言 5:Py ...

  4. Python面试题之Python面向对象编程汇总

    面向对象的设计思想是从自然界中来的,因为在自然界中,类(Class)和实例(Instance)的概念是很自然的.Class是一种抽象概念,比如我们定义的Class——Student,是指学生这个概念, ...

  5. python面试题二:Python 基础题

    1.位和字节的关系? Byte 字节 bit 位 1Byte = 8bit 2.b.B.KB.MB.GB 的关系? 1Byte = 8bit KB 1KB=1024B MB 1MB=1024KB GB ...

  6. python面试题之Python支持什么数据类型?

    所属网站分类: 面试经典 > python 作者:外星人入侵 链接:http://www.pythonheidong.com/blog/article/67/ 来源:python黑洞网,专注py ...

  7. python面试题之Python如何实现单例模式?

    #使用__metaclass__(元类)的高级python用法 class Singleton2(type): def __init__(cls, name, bases, dict): super( ...

  8. python面试题三:Python 网络编程与并发

    1 简述 OSI 七层协议. OSI七层协议模型主要是: 应用层(Application):为用户的应用程序(例如电子邮件.文件传输和终端仿真)提供网络服务. 表示层(Presentation):使用 ...

  9. Python面试题之python是一种什么语言及优缺点

    1.说说python是一种什么语言? 参考答案:python是一门动态解释性的强类型定义语言 编译型vs解释型 编译型优点:编译器一般会有预编译的过程对代码进行优化.因为编译只做一次,运行时不需要编译 ...

随机推荐

  1. webpack文档翻译

    https://segmentfault.com/a/1190000007568507

  2. Bellman-Ford算法(有向图)

    #include <iostream> #include <cstring> #include <cstdio> #define MAX 100 #define I ...

  3. Laravel5.1 模型 --远层一对多关系

    远层一对多我们可以通过一个例子来充分的了解它: 每一篇文章都肯定有并且只有一个发布者 发布者可以有多篇文章,这是一个一对多的关系.一个发布者可以来自于一个国家 但是一个国家可以有多个发布者,这又是一个 ...

  4. JAVA大数(转)

    1.输入 首先要想输入需要先包括: import java.util.*; 我们需要其中的 Scanner类声明的对象来扫描控制台输入. 针对A+B来说: import java.util.*; pu ...

  5. Store update, insert, or delete statement affected an unexpected number of rows (0). Entities may have been modified or deleted since entities were loaded.

    EF6进行Insert操作的时候提示错误 Store update, insert, or delete statement affected an unexpected number of rows ...

  6. Entity Framework查询生成大量的子查询,如何避免?求救

    最近使用Entity Framework做一个中型的项目,一张表含有千万条数据,并没有使用很复杂的查询,只是程序上使用了DTO进行帅选数据,且使用了分页,效果很不理想.经过跟踪sql,我发现很多简单的 ...

  7. sql优化(2)

    转自:https://www.cnblogs.com/Jacck/p/8030455.html 看到一篇非常全面的SQL优化文章,在开发的工作中往往不考虑性能上的缺失(在一开始的时候数据量不大也看不出 ...

  8. C# .net 数组倒序排序

    1.数组方法 Array.Sort(Array Array);  此方法为数组的排序(正序)方法 Array.Reverse(Array Array);  此方法可以将数组中的值颠倒 两个方法结合使用 ...

  9. IOS数据持久化之归档NSKeyedArchiver, NSUserDefaults,writeToFile

    //2.文件读写 //支持:NSString, NSArray , NSDictionay, NSData //注:集合(NSArray, NSDictionay)中得元素也必须是这四种类型, 才能够 ...

  10. 总结! http post请求 application/x-www-form-urlencoded body体数据获取不到?

    首先,简单介绍下Http请求中Content-Type类型 类型格式:type/subtype(;parameter)? type 主类型,任意的字符串,如text,如果是*号代表所有: subtyp ...