• python中的__init__ 、__new__、__call__小结


    这篇文章主要介绍了python中的__init__ 、__new__、__call__小结,需要的朋友可以参考下

    1.__new__(cls, *args, **kwargs)  创建对象时调用,返回当前对象的一个实例;注意:这里的第一个参数是cls即class本身
    2.__init__(self, *args, **kwargs) 创建完对象后调用,对当前对象的实例的一些初始化,无返回值,即在调用__new__之后,根据返回的实例初始化;注意,这里的第一个参数是self即对象本身【注意和new的区别】
    3.__call__(self,  *args, **kwargs) 如果类实现了这个方法,相当于把这个类型的对象当作函数来使用,相当于 重载了括号运算符
     

    看具体的例子:

    复制代码 代码如下:

    class O(object):

        def __init__(self, *args, **kwargs):
            print "init"
            super(O, self).__init__(*args, **kwargs)

        def __new__(cls, *args, **kwargs):
            print "new", cls
            return super(O, cls).__new__(cls, *args, **kwargs)

        def __call__(self,  *args, **kwargs):
            print "call"
          

        oo = O()
        print "________"
        oo() 


    打印出来的是:
    复制代码 代码如下:

    new
    init
    ________
    call

    比如:Python Singleton(单例模式)实现,那我们是不是只是重载一些__new__方法就可以了
    复制代码 代码如下:

    class Singleton1(object):
        """ 重载new方法"""
        def __new__(cls, *args, **kwargs):
            if not "_instance" in vars(cls):
                cls._instance = super(Singleton1, cls).__new__(cls, *args, **kwargs)
            return cls._instance

    可不可以重载__init__方法呢?明显不可以,因为__init__之前调用了__new__方法,这时候已经生成了一个对象了,没办法实现单例模式

    ===========================================
     

    注意1、__init__并不相当于C#中的构造函数,执行它的时候,实例已构造出来了。

    1
    2
    3
    4
    5
    class A(object):
        def __init__(self,name):
            self.name=name
        def getName(self):
            return 'A '+self.name

    当我们执行

    1
    a=A('hello')

    时,可以理解为

    1
    2
    a=object.__new__(A)
    A.__init__(a,'hello')

    即__init__作用是初始化已实例化后的对象。

    注意2、子类可以不重写__init__,实例化子类时,会自动调用超类中已定义的__init__

    1
    2
    3
    4
    5
    6
    7
    class B(A):
        def getName(self):
            return 'B '+self.name
     
    if __name__=='__main__':
        b=B('hello')
        print b.getName()

    但如果重写了__init__,实例化子类时,则不会隐式的再去调用超类中已定义的__init__

    1
    2
    3
    4
    5
    6
    7
    8
    9
    class C(A):
        def __init__(self):
            pass
        def getName(self):
            return 'C '+self.name
     
    if __name__=='__main__':
        c=C()
        print c.getName()

    则会报"AttributeError: 'C' object has no attribute 'name'”错误,所以如果重写了__init__,为了能使用或扩展超类中的行为,最好显式的调用超类的__init__方法

    1
    2
    3
    4
    5
    6
    7
    8
    9
    class C(A):
        def __init__(self,name):
            super(C,self).__init__(name)
        def getName(self):
            return 'C '+self.name
     
    if __name__=='__main__':
        c=C('hello')   
        print c.getName()
     
     
  • 相关阅读:
    linux命令df中df -h和df -i的区别
    linux系统df和du命令的区别
    Linux type命令的用法
    《DNS的正向反向解析》RHEL6
    《DNS服务缓存的建立》RHEL6
    《服务器的追踪与审计》RHEL6
    《LDAP服务器和客户端的加密认证》RHEL6——第二篇 运维工程师必考
    《ISCSI集中存储》RHEL6——CE
    《LDAP服务器的配置与客户端的测试》RHEL6——第一篇 运维工程师必考
    《RHEL6硬盘的分区和swap分区管理》——硬盘分区的大总结
  • 原文地址:https://www.cnblogs.com/kungfupanda/p/5237255.html
Copyright © 2020-2023  润新知