|
发表于 2017-4-2 09:36:50
|
显示全部楼层
class A(object):
def __init__(self):
print "init"
def __new__(cls,*args, **kwargs):
print "new %s"%cls
return object.__new__(cls, *args, **kwargs)
A()
输出:
new <class '__main__.A'>
init
知识点:
继承自object的新式类才有__new__
__new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供
__new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例
__init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上可以完成一些其它初始化的动作,__init__不需要返回值
若__new__没有正确返回当前类cls的实例,那__init__是不会被调用的,即使是父类的实例也不行
class A(object):
pass
class B(A):
def __init__(self):
print "init"
def __new__(cls,*args, **kwargs):
print "new %s"%cls
return object.__new__(A, *args, **kwargs)
b=B()
print type(b)
输出:
new <class '__main__.B'>
<class '__main__.A'>
与__init__一样,每当实例化类时MyClass(*args, **kwargs) ,__new__ 都会默认自动执行MyClass.__new__(*args, **kwargs),如果想改变默认__new__行为,可以对它自定义
看一个英寸转换为米例子:
1 class inch(float):
2 def __new__(cls, arg=0.0):
3 return float.__new__(cls, arg * 0.0254)
改变了__new__的自定义实例化类
inch继承自float
如果同时存在__new__ ,__init__则__new__比__init__先运行,__new__将返回一个对象
a = inch(12)
print isinstance(a, float) 这里将会是True,可以认为inch实例化的对象类型是float,是可以直接拿来做加减乘除等操作
print a + 0.1 #进行基本的浮点运算。
再来个单例的,通过重载__new__实现单例:
class Singleton(object):
def __new__(cls, *args, **kwargs):
if not hasattr(cls, '_instance'):
cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
return cls._instance |
|