• 技术文章 >Python技术 >Python高级

    一文读懂Python中__new__和__init__的区别与联系

    silencementsilencement2019-07-19 09:29:16原创2987

    __new__ 和 __init__ 的区别主要表现在:

    它自身的区别;

    在Python中新式类和老式类的定义。

    __new__ 负责对象的创建,而 __init__ 负责对象的初始化。

    __new__:创建对象时调用,会返回当前对象的一个实例

    __init__:创建完对象后调用,对当前对象的一些实例初始化,无返回值

    1. 在类中,如果__new__和__init__同时存在,会优先调用__new__

    class ClsTest(object):
        def __init__(self):
            print("init")
        def __new__(cls,*args, **kwargs):
            print("new")
      
    ClsTest()

    输出

    new

    2. 如果__new__返回一个对象的实例,会隐式调用__init__

    代码实例:

    class ClsTest(object):
        def __init__(self):
            print ("init")
        def __new__(cls,*args, **kwargs):
            print ("new %s"%cls)
            return object.__new__(cls, *args, **kwargs)
      
    ClsTest()

    输出

    new <class '__main__.ClsTest'>
    init

    3. __new__方法会返回所构造的对象,__init__则不会。__init__无返回值。

    class ClsTest(object):
         def __init__(cls):
                 cls.x = 2
                 print ("init")
                 return cls
      
    ClsTest()

    输出

    init
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    TypeError: __init__() should return None, not 'ClsTest'

    4. 若__new__没有正确返回当前类cls的实例,那__init__是不会被调用的,即使是父类的实例也不行

    class ClsTest1(object):
        pass
      
    class ClsTest2(ClsTest1):
        def __init__(self):
            print ("init")
        def __new__(cls,*args, **kwargs):
            print ("new %s"%cls)
            return object.__new__(ClsTest1, *args, **kwargs)
      
    b=ClsTest2()
    print (type(b))

    输出

    new <class '__main__.ClsTest2'>
    <class '__main__.ClsTest1'>

    总结

    1. 继承自object的新式类才有__new__

    2. __new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供,__new__必须要有返回值,返回实例化出来的实例,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例

    3. __init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上可以完成一些其它初始化的动作,__init__不需要返回值

    4. 如果__new__返回一个对象的实例,会隐式调用__init__。

    专题推荐:new init
    品易云
    上一篇:深入理解python中的yield 下一篇:手把手教你用Python PIL模块随机生成中文验证码

    相关文章推荐

    • round四舍五入在python2与python3版本间区别• 详细分析Python中运算符"=="和"is"的差别• Python中eval与exec的使用及区别

    全部评论我要评论

    © 2021 Python学习网 苏ICP备2021003149号-1

  • 取消发布评论
  • 

    Python学习网