python高级编程之超类02:super的缺陷

# -*- coding: utf-8 -*-

# python:2.x

__author__ = ‘Administrator‘

#当使用多重继承层次结构时,再使用super的时候是非常危险的,主要是因为类的初始化,基类不在__init__中被隐式调用

#1滥用super和传统调用

#来自james knight(http://funm.net/super-harmful)示例中,类C使用__init__方法调用其基类,这样类B被调用2次

class A(object):

def __init__(self):

print ‘A‘

super(A,self).__init__()

class B(object):

def __init__(self):

print ‘B‘

super(B,self).__init__()

class C(A,B):

def __init__(self):

print ‘C‘

A.__init__(self)

B.__init__(self)

print ‘MRO:‘,[x.__name__ for x in C.__mro__]#MRO: [‘C‘, ‘A‘, ‘B‘, ‘object‘]

C()

"""

C

A

B

B

------------

出现这样的情况是:C实例调用 了A.__init__(self),因而super(A,self).__init__()将调用B构造程序,换句话说,super应该被乃至整个类层次中

问题是有时候这样的层次结构一部分位于第三方代码中,许多由多重继承引入层次调用相关缺陷都可以在JAMES页面上找到,为了避免这样问题,应该问题是在

子类化之前看看__mro__特性,如果它不存在,处理就是一个旧式类,避免使用super可能更安全一些

如下:

"""

from SimpleHTTPServer import SimpleHTTPRequestHandler

#print SimpleHTTPRequestHandler.__mro__AttributeError: class SimpleHTTPRequestHandler has no attribute ‘__mro__‘

"""

如果__mro__存在,则快速地看看每个mro所涉及的类的构造程序代码,如果到处都使用super,那非常好,也可以使用它,否则就试着保持一致性

collections.deque能够被安全地子类化,就可以使用super,因为它直接子类化了object

"""

from collections import deque

print deque.__mro__#(<type ‘collections.deque‘>, <type ‘object‘>)

#randeom.Random是一个存在于_random模块中另一个类封装器

from random import Random

print Random.__mro__#(<class ‘random.Random‘>, <type ‘_random.Random‘>, <type ‘object‘>)

#Zope类,

"""

from zope.app.container.brower.adding import Adding as s

s.__mro__

官方:http://www.zope.com/

有兴趣的朋友可以浏览下

"""

#不同种类参数

#super用法另一个问题是初始化中参数传递,类在没有相同签名的情况下怎么调用其基类的__init__代码呢?

class Bases(object):

def __init__(self):

print ‘bases‘

super(Bases,self).__init__()

class B(Bases):

def __init__(self):

print ‘B‘

super(B,self).__init__()

class B1(Bases):

def __init__(self):

print ‘b1‘

super(B1,self).__init__()

class MyClass(B,B1):

def __init__(self,arg):

print ‘my class arg‘

super(MyClass,self).__init__(arg)

#m=MyClass(10)TypeError: __init__() takes exactly 1 argument (2 given)

#解决方法之下是使用*args,**kw魔法,所有构造程序将传递所有参数,即使不使用它们

class Bb(object):

def __init__(self,*args,**kw):

print ‘b1‘

super(Bb,self).__init__(*args,**kw)

class bb(Bb):

def __init__(self,*args,**kw):

print ‘bb‘

super(bb,self).__init__(*args,**kw)

class bbb(Bb):

def __init__(self,*args,**kw):

print ‘bbb‘

super(bbb,self).__init__(*args,**kw)

class MyclSS(bb,bbb):

def __init__(self,arg):

print ‘myclss‘

super(MyclSS,self).__init__(arg)

mm=MyclSS(10)

"""

myclss

bb

bbb

b1

"""

"""

但是这样是一个很糟糕的修复方法,因为它使所有构造程序将接受任何类型参数,导致代码变得很脆弱,因为任何参数被传递并且通过,另一种解决方法是在yclSS

中使用经典的__init__调用,但是这将会导致产生第一种缺陷

"""

python高级编程之超类02:super的缺陷,布布扣,bubuko.com

时间: 2024-10-18 01:08:16

python高级编程之超类02:super的缺陷的相关文章

python高级编程之访问超类中的方法:super()

# -*- coding: utf-8 -*- # python:2.x __author__ = 'Administrator' #超类01 #它是一个内建类型,用于访问属于某个对象超类特性 print super#<type 'super'> #如果已习惯于通过直接调用父类将self作为第一参数来访问的特性,可能会出现混乱 #经典方法 class M(object): def s(self): print u'不是超类方法' class S(M): def s(self): M.s(sel

python高级编程之装饰器02

#装饰器02 #参数检查 #主要是用在接收或者返回函数,在特定上下文执行时可能有用 #例如:有一个函数通过XML-RPC调用,python将不能和静态类语言中一样直接提供它的完整签名,当XML-RPC客户要求函数签名时,就需要这样的能力 """ xml-rpc相关学习:http://zh.wikipedia.org/wiki/XML-RPC """ #装饰器能提供这种签名类型,并确保输入输出与此有关,如下 from itertools impor

Python高级编程pdf

下载地址:网盘下载 内容简介  · · · · · · <Python高级编程>通过大量的实例,介绍了Python语言的最佳实践和敏捷开发方法,并涉及整个软件生命周期的高级主题,诸如持续集成.版本控制系统.包的发行和分发.开发模式.文档编写等.<Python高级编程>首先介绍如何设置最优的开发环境,然后以Python敏捷开发方法为线索,阐述如何将已被验证的面向对象原则应用到设计中.这些内容为开发人员和项目管理人员提供了整个软件工程中的许多高级概念以及专家级的建议,其中有些内容的意义

python高级编程之(类级):子类内建类型

# -*- coding: utf-8 -*- # python:2.x __author__ = 'Administrator' #类级 #在2.2中,提出了类型(type0与类(class)统一(请访问:https://www.python.org/download/releases/2.2.3/descintro(可能已经不存在了))-这使内建类型的子类化成为可能,并且添加一个新内建类型object #用于所有内建类的公共祖先 #展示一个名为distinctdict类的代码,与平常的dic

python高级编程:有用的设计模式1

# -*- coding: utf-8 -*-__author__ = 'Administrator'#python高级编程:有用的设计模式#设计械是可复用的,某种程序上它对软件设计中觉问题提供的语言相关解决识方案,最近最流行的书籍:"""gamma.heim.johson和vlissides a.k.a"四人组(gof)"编写的elements of reusable object-oriented software(中文:<设计模式:可复用面向对

python高级编程之最佳实践,描述符与属性01

# -*- coding: utf-8 -*- # python:2.x __author__ = 'Administrator' #最佳实践 """ 为了避免前面所有的问题,必须考虑到几下几点: ~1:应该避免多重继承,可以一些设计模式来代替它 ~2:super使用必须一致,在类层次结构中,应该在所有地方都使用super或者彻底不使用它,滥用super和传统调用是一种滥用方法,建议使用super ~3:不要滥用经典类和新式类,两者都具备代码库将导致不同的mro表现 ~4:调

python高级编程:有用的设计模式2

# -*- coding: utf-8 -*- __author__ = 'Administrator' #python高级编程:有用的设计模式 #代理 """ 代理对一个代价昂贵或者远程的资源提供了一个非直接访问的机制 在客户和主意之间,如图.它用来优化对高代价主题的访问,比如,在前一章中描述的memoize装饰器可以被认为是一个代理 ,它还可以用提供到一个主题智能访问,例如,大的视频文件可以封闭在代理中,以避免在用户仅仅请教其标题时就将文件载入到内存中 urllib2出给

python高级编程之装饰器04

from __future__ import with_statement # -*- coding: utf-8 -*- # python:2.x __author__ = 'Administrator' #with和contextlib #对于要确保即使发生一个错误时也能运行一些清理代码而言,try...finally语句很有用,对以下场景,如: """ 关闭一个文件, 释放一个锁 创建一个临时代码补丁 在特殊环境中运行受保护代码 ----------- with语句覆盖

python高级编程之迭代器与生成器

# -*- coding: utf-8 -*- # python:2.x __author__ = 'Administrator' #迭代器与生成器 #--------------------------------------- #迭代器基于2个方法 """ next:返回容器下一个项目 __iter__:返回迭代器本身 """ #通过内建函数和序列来创建 i=iter('abc') print i.next()#a print i.next(