1. 为什么只说方法不说属性

关于“子类对象如何访问父类的同名属性“是没有意义的。因为父类的属性子类都有,子类还有父类没有的属性,在初始化时,给子类对象具体化所有的给定属性,完全没必要访问父类的属性,因为是一样的。而访问同名方法就不一样了,因为子类会重写父类的同名函数。

2、子类访问父类的两种方法

假设Base为基类

class Base(object):

def __init__(self):

print “Base init”

2.1 普通方法(调用父类的未绑定的构造方法)

在调用一个对象的方法时,该方法的self参数会被自动绑定到对象上(称为绑定方法)。但如果直接调用类的方法(比如A.__init),那么就没有实例会被绑定。这样就可以自由的提供需要的self参数,这种方法称为未绑定方法。

class Leaf(Base):

def __init__(self):

Base.__init__(self)

print ("Leaf init")

2.2 super方法

class Leaf(Base):

def __init__(self):

super(Leaf, self).__init__() # 在单继承中等价于super().__init__()

print “Leaf init”

2.3 上述两个方法的结果

>>>leaf = Leaf()

"Base init"

"Leaf init"

在Python3中的类都是新式类,广度优先的查找顺序,在定义一个类时就会生成一个MRO列表(经典类没有MRO列表,深度优先),查找顺序就是按照这个列表中的类的顺序从左到右进行的。

3、钻石继承中的问题

”钻石继承“指的是一个子类继承自多个父类,成钻石形状。

3.1 如何解决钻石继承中父类被多次初始化

C++使用虚拟继承来解决钻石继承问题;Java禁止使用多继承;Ruby禁止使用多继承;Python和C++一样,支持多继承的语法。但Python的解决思路和C++完全不一样,Python是的用就是super。所以,我们有必要来详细探讨super的机制。

我把第二章的代码用super重写一下,使用如下的继承关系

class Base(object):

def __init__(self):

print ("Base init")

class Medium1(Base):

def __init__(self):

super(Medium1, self).__init__()

print ("Medium1 init")

class Medium2(Base):

def __init__(self):

super(Medium2, self).__init__()

print ("Medium2 init")

class Leaf(Medium1, Medium2):

def __init__(self):

super(Leaf, self).__init__()

print ("Leaf init")

>>> leaf = Leaf()

"Base init"

"Medium2 init"

"Medium1 init"

"Leaf init"

可以看到整个初始化过程符合我们的预期,Base只被初始化了1次。而且重要的是,相比原来的普通写法,super方法并没有写额外的代码,也没有引入额外的概念.

3.2 super的内核:MRO

要理解super的原理,就要先了解mro。mro是method resolution order的缩写,表示了类继承体系中的成员解析顺序。

在python中,每个类都有一个mro的类方法。我们来看一下钻石继承中,Leaf类的mro是什么样子的:

>>>Leaf.mro()

[Leaf, Medium1, Medium2, Base]

可以看到mro方法返回的是一个祖先类的列表。Leaf的每个祖先都在其中出现一次,这也是super在父类中查找成员的顺序。

通过mro,python巧妙地将多继承的图结构,转变为list的顺序结构。super在继承体系中向上的查找过程,变成了在mro中向右的线性查找过程,任何类都只会被处理一次。

通过这个方法,python解决了多继承中的2大难题:

查找顺序问题。从Leaf的mro顺序可以看出,如果Leaf类通过super来访问父类成员,那么Medium1的成员会在Medium2之前被首先访问到。如果Medium1和Medium2都没有找到,最后再到Base中查找。

钻石继承的多次初始化问题。在mro的list中,Base类只出现了一次。事实上任何类都只会在mro list中出现一次。这就确保了super向上调用的过程中,任何祖先类的方法都只会被执行一次。

至于mro的生成算法,可以参考这篇wiki:C3算法

3.3 super的具体用法

首先看一下super的官方文档,重点讲第一种和第三种的用法,因为第二种用处不大

Help on class super in module __builtin__:

class super(object)

| super(type, obj) -> bound super object; requires isinstance(obj, type)

| super(type) -> unbound super object

| super(type, type2) -> bound super object; requires issubclass(type2, type)

3.2.1 super(type,obj)

这种写法要从右往左的分析参数:obj告诉我们self绑定的对象所属类的MRO顺序,type告诉我们从MRO顺序的哪个位置开始找

当我们在Leaf的init中这样写super时,super(Leaf, self).init()的意思是说:

获取self所属类的mro, 也就是[Leaf, Medium1, Medium2, Base]

从mro中Leaf右边的一个类开始,依次寻找__init__函数。这里是从Medium1开始寻找

一旦找到,就把找到的__init__函数绑定到self对象,并返回

class Leaf(Medium1, Medium2):

def __init__(self):

super(Leaf, self).__init__()

print “Leaf init”

从这个执行流程可以看到,如果我们不想调用Medium1的__init__,而想要调用Medium2的__init__,那么super应该写成:super(Medium1, self).__ init __()

3.2.2 super(type,type2)

当我们在Leaf中写类方法的super时:

class Leaf(Medium1, Medium2):

def __new__(cls):

obj = super(Leaf, cls).__new__(cls)

print “Leaf new”

return obj

super(Leaf, cls).new(cls)的意思是说:

获取cls这个类的mro,这里也是[Leaf, Medium1, Medium2, Base]

从mro中Leaf右边的一个类开始,依次寻找__new__函数

一旦找到,就返回“非绑定”的__new__函数

3.2.3 super一些补充的知识点

super实现原理:通过c3算法,生成mro(method resolution order)列表,根据列表中元素顺序查询调用。新式类调用顺序为广度优先,旧式类为深度优先,可以通过__mro__来查看。

python2没有默认继承object;python3默认全部继承自object类,都是新式类

如果子类继承了多个父类,它只需要使用一次super函数就可以

4. 子类如何访问父类的同名方法

讲了这么多,把上面的知识点综合起来,我们就可以解决子类如何访问父类的同名方法。

看下面的代码:

class A:

def f_a(self):

print('--------A--------')

class B:

def f_a(self):

print('-------B-------')

class C(A,B):

def f_a(self):

print('--------C--------')

c = C()

c.f_a()

》》》--------C--------

如果我们想在C类中的f_a方法里面使用父类A或者父类B的方法该如何操作呢

方法有两种:

4.1 方法一:调用父类的未绑定方法

class C(A,B):

def f_a(self):

A.f_a(self)

B.f_a(self)

print('--------C--------')

c = C()

c.f_a()

》》》--------A--------

》》》-------B-------

》》》--------C--------

这里调用父类的f_a方法时括号里面要写self,表明这是一个类调用。

缺点:比如说如果修改了父类的名称,那么在子类中会涉及多出修改,并且python是允许多继承的语言,上述方法在多继承时就要重复写多次,显得累赘,为了解决这些问题,python引进了super()机制。

4.2 super机制

class C(A,B):

def f_a(self):

super().f_a()

print('--------C--------')

c = C()

c.f_a()

》》》--------A--------

》》》--------C--------

这里直接使用super()方法会调用A类的f_a方法,因为它会默认多继承中从左到右的顺序来调用,那么有人就会问了,假如我想调用B类中的f_a方法是不是把A和B对调下呢,这种方法也行,不过这不算的上是一种较为巧妙的方法。

我们还可以使用super()方法,但要对其修改下:

class C(A,B):

def f_a(self):

super(C,self).f_a()

super(A,self).f_a()

print('--------C--------')

c = C()

c.f_a()

》》》--------A--------

》》》-------B-------

》》》--------C--------

super(C,self).f_a()会调用最左边的(即A类)的f_a方法,而super(A,self).f_a()会调用A类后面那个类的f_a()方法,若果继承的不止两个类,如果要调用某个类方法,只要知道前一个类名就可以调用。

python重点知识 钻石_python——子类对象如何访问父类的同名方法相关推荐

  1. C++子类对象隐藏了父类的同名成员函数(隐藏篇)

    https://blog.csdn.net/alpha_love/article/details/75222175 #include <iostream> #include <std ...

  2. 这或许是你见过的最全面的 Python 重点知识总结

    这是一份总结 Python 重点知识点点文章. 由于有太多的东西,本篇点篇幅有点长,这也是我"缝缝补补"总结了好久的东西,强烈建议收藏.关注.点赞,再慢慢看~ [注]完整版代码.数 ...

  3. 浅谈将子类对象赋值给父类对象

    最近对将子类对象赋值给父类对象有点心得,想和大家分享一下,但本人水平有限,请各位指正和批评.言归正传,下面是几个小例子,请大家看一看. 测试一 父类: public class Supclass {p ...

  4. 子类调用父类的同名方法和属性

    Python面向对象中,利用子类调用父类的同名方法和属性. 首先我们定义了两个父类,Master和School # @author: 北山啦 # FileName: 12子类调用父类的同名方法 # @ ...

  5. 最全面的Python重点知识汇总,建议收藏!

    点击上方"视学算法",选择加"星标"或"置顶" 重磅干货,第一时间送达 来源丨网络,由法纳斯特整理 这是一份来自于 SegmentFault ...

  6. 你见过的最全面的Python重点知识总结!

    这是一份来自于 SegmentFault 上的开发者 @二十一 总结的 Python 重点.由于总结了太多的东西,所以篇幅有点长,这也是作者"缝缝补补"总结了好久的东西,强烈建议收 ...

  7. 最全面的Python重点知识汇总,建议码住

    这是一份来自于 SegmentFault 上的开发者 @二十一 总结的 Python 重点.由于总结了太多的东西,所以篇幅有点长,这也是博主"缝缝补补"总结了好久的东西. Py2 ...

  8. 你见过的最全面的Python重点知识汇总

    原文链接: https://segmentfault.com/a/1190000018737045 这是一份来自于 SegmentFault 上的开发者 @二十一 总结的 Python 重点.由于总结 ...

  9. python入门知识大全_python基础知识有哪些

    Python基础知识:1.引用和对象:2.可变数据类型和不可变数据类型:3.引用传递和值传递:4.深拷贝和浅拷贝:5.基本数据类型:6.关键字.标识符和内置函数:7.算术.赋值运算符等等. Pytho ...

最新文章

  1. activity切换交互动画
  2. imx51-linux的cpuinfo之分析
  3. EOS从入门到精通-账户体系(文字稿)
  4. 原生js实现JSON.parse()和JSON.stringify()
  5. 设置Apache Hadoop多节点集群
  6. 解决Office系列安装不上的办法
  7. java堆外内存溢出_JVM 案例 - 堆外内存导致的溢出错误
  8. Apache Zookeeper 集群环境搭建
  9. HTTPBrowserCapabilities---在asp.net中显示浏览器属性
  10. sqlserver 两个数值字段相加_SQLServer 中多行数据合并成一行数据(一个字段)
  11. 使用mvc2里面的area让网站更有条理
  12. [jQuery]使用jQuery.Validate进行客户端验证(初级篇)——不使用微软验证控件的理由...
  13. 深入理解JVM虚拟机读书笔记——垃圾回收算法
  14. mysql开启远程登录
  15. Linux下 python调用讯飞离线语音合成(tts)
  16. MySQL服务器的超级管理员名称是_mysql服务器忘记了超级管理员root密码
  17. 求A,B 输入三位数字N,求两位数AB(其中个位数字为B,十位数字为A,且有0 #60 A #60 B #60 9)。
  18. 美团校招题 小团的装饰物
  19. c语言代码怎样制作成一个游戏?
  20. 把握出租车行驶的数据脉搏 :出租车轨迹数据给你答案!

热门文章

  1. Angular自学笔记(?)ViewChild和ViewChildren
  2. dotnet不是内部或外部的命令,也不是可运行的程序或批处理文件
  3. EasyWeChat微信开放平台第三方平台接入
  4. 编写一个简单的spring MVC程序
  5. linux性能分析常用的命令
  6. easyui datagrid 列拖动
  7. YTU 2924: 文件操作--二进制文件读入
  8. CSS3中border-radius、box-shadow与gradient那点事儿
  9. win32 api 文件操作!
  10. 免安装Mysql在Mac中的神坑之Access denied for user 'root'@'localhost' (using password: YES)