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

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()方法,若果继承的不止两个类,如果要调用某个类方法,只要知道前一个类名就可以调用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/258621.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android-Universal-Image-Loader 的使用说明

这个图片异步载入并缓存的类已经被非常多开发人员所使用,是最经常使用的几个开源库之中的一个,主流的应用,随便反编译几个火的项目,都能够见到它的身影。但是有的人并不知道怎样去使用这库怎样进行配置,网上查到的信息…

faster rcnn end2end 训练与测试

除了前面讲过的rpn与fast rcnn交替训练外,faster rcnn还提供了一种近乎联合的训练,姑且称为end2end训练。 根据论文所讲,end2end的训练一气呵成,对于前向传播,rpn可以作为预设的网络提供proposal.而在后向传播中&…

jquery ui动态切换主题的一种实现方式

这两天看coreservlets上的jQuery教程&#xff0c;虽然比较老了&#xff0c;不过讲得还是不错。最后一部分讲jQuery ui 主题切换&#xff0c;用他介绍的方法实现不了。于是自己修改了下&#xff0c;可以了。代码如下&#xff1a;html部分&#xff1a;<fieldset class"ui…

[学习总结]7、Android AsyncTask完全解析,带你从源码的角度彻底理解

我们都知道&#xff0c;Android UI是线程不安全的&#xff0c;如果想要在子线程里进行UI操作&#xff0c;就需要借助Android的异步消息处理机制。之前我也写过了一篇文章从源码层面分析了Android的异步消息处理机制&#xff0c;感兴趣的朋友可以参考 Android Handler、Message完…

python字频统计软件_python结巴分词以及词频统计实例

python结巴分词以及词频统计实例发布时间&#xff1a;2018-03-20 14:52,浏览次数&#xff1a;773, 标签&#xff1a;python# codingutf-8Created on 2018年3月19日author: chenkai结巴分词支持三种分词模式&#xff1a;精确模式: 试图将句子最精确地切开&#xff0c;适合文…

html从入门到卖电脑(三)

CSS3中和动画有关的属性有三个 transform、 transition 和 animation。下面来一一说明: transform 从字面来看transform的释义为改变&#xff0c;使…变形&#xff1b;转换 。这里我们就可以理解为变形。那都能怎么变呢&#xff1f; none 表示不进行变换&#xff1b; rotat…

visual studio 2015安装 无法启动程序,因为计算机丢失D3DCOMPILER_47.dll 的解决方法

对于题目中的解决方法&#xff0c;我查到了微软提供的一个方案&#xff1a;https://support.microsoft.com/en-us/help/4019990/update-for-the-d3dcompiler-47-dll-component-on-windows 进入如下页面&#xff1a;http://www.catalog.update.microsoft.com/Search.aspx?qKB4…

UI1_UIView层操作

// // ViewController.m // UI1_UIView层操作 // // Created by zhangxueming on 15/7/1. // Copyright (c) 2015年 zhangxueming. All rights reserved. //#import "ViewController.h"interface ViewController ()endimplementation ViewController- (void)view…

JavaScript Patterns 1 Introduction

1.1 Pattern "theme of recurring events or objects… it can be a template or model which can be used to generate things" (http://en.wikipedia.org/wiki/Pattern). • Design patterns - Elements of Reusable Object-Oriented Software. • Coding patte…

基于像素聚类的分割方法基于slic的方法_博士论文摘要 | 张荣春:数码影像与TLS点云数据融合提取地质结构面方法研究...

《测绘学报》构建与学术的桥梁 拉近与权威的距离数码影像与TLS点云数据融合提取地质结构面方法研究张荣春1,21.南京邮电大学地理与生物信息学院, 江苏 南京 210023;2.河海大学地球科学与工程学院, 江苏 南京 211100收稿日期&#xff1a;2019-03-27基金项目&#xff1a;国家自然…

制作IOS 后台极光推送时,遇到的小问题

推送广义上分为两种&#xff0c; 一种是 程序在前台的时候&#xff0c;不想在任务栏里面显示通知&#xff0c;直接在app中进行某种操作。这个叫做自定义消息。这个是在前台时&#xff0c;app与极光后台建立了一个长链接。 另一种是 程序处于前、后台 或者杀死状态的时候&…

Visual Studio 2008 环境变量的配置(参考设置VS2010)

本文转载自&#xff1a;http://blog.csdn.net/tracyliang223/article/details/21539361COPY FROM&#xff1a;http://www.cnblogs.com/waterlin/archive/2011/10/31/2230341.html 在调试 Visual Studio 2008 程序时&#xff0c;经常有一些动态链接库&#xff08;即 dll 文件&am…

Linq 中 Any与All

昨天突然看到之前写的一个积累文档&#xff0c;其中文档中有一个Linq Any和All的注意事项&#xff1a;注意Any 和 All var list new List<int>(); var aa list.All(n > n > 1); var bb list.Any(n > n > 1); // aa: true bb: false其中List是一个元…

jaxb转xml空值双标签_单品运营思维:标签-词路-聚焦-直搜-超直

非标品标签思维&#xff1a;针对非标品 主要是2.0为主的打法根据搜索入池的关键词&#xff0c;有什么词做什么词。有个细节&#xff1a;不一定进什么词做什么词&#xff0c;这个维度当中加入3.0的思维3.0入手 转2.0再切3.0(检测词路健康度&#xff0c;非严格意义估算单量)举例&…

如何在PFSense中设置故障转移和负载平衡

故障转移是一种备份操作模式&#xff0c;仅在主系统由于系统故障或任何计划停机时间而变得不可用时&#xff0c;系统组件&#xff08;如网络&#xff09;的操作才由辅助系统承担。在本教程中&#xff0c;我们将看到如何设置故障转移和负载平衡&#xff0c;以使pfSense能够将流量…

图像金字塔总结

本文转载自&#xff1a; http://blog.csdn.net/dcrmg/article/details/52561656 一、 图像金字塔 图像金字塔是一种以多分辨率来解释图像的结构&#xff0c;通过对原始图像进行多尺度像素采样的方式&#xff0c;生成N个不同分辨率的图像。把具有最高级别分辨率的图像放在底部…

表单的get和post使用情景

GET和POST两种方法都是将数据送到服务器&#xff0c;但你该用哪一种呢&#xff1f;HTTP标准包含这两种方法是为了达到不同的目的。POST用于创建资源&#xff0c;资源的内容会被编入HTTP请示的内容中。例如&#xff0c;处理订货表单、在数据库中加入新数据行等。 当请求无副作用…

什么叫做罗列式_项目起盘的时候,如何确定自己该做什么社群?

这是祁杰『社群日记』第48篇持续日更&#xff0c;做最懂社群的营销咨询人很多人手上有资源&#xff0c;准备起盘项目的时候&#xff0c;总会思考一个问题&#xff1a;我能做什么样的社群&#xff1f;今天我们从用户需求出发&#xff0c;拆解一下哪些社群是能够确切地满足用户的…

C++ exit 与 return 浅析

【摘要】 本文从代码形式。经常使用方式&#xff0c;相关概念&#xff0c;调用关系和比較分析&#xff0c;这5个维度浅析 exit 与 return 在C的同样点与差别。【常见形式】 exit(0)&#xff1a; 正常执行程序并退出程序。 exit(1)&#xff1a; 非正常执行导致退出程序&…

Feature Pyramid Networks for Object Detection 总结

最近在阅读FPN for object detection,看了网上的很多资料&#xff0c;有些认识是有问题的&#xff0c;当然有些很有价值。下面我自己总结了一下&#xff0c;以供参考。 1. FPN解决了什么问题&#xff1f; 答&#xff1a; 在以往的faster rcnn进行目标检测时&#xff0c;无论…