Python中的__init__和__new__
__init__ 方法通常用在初始化一个类实例的时候,但__init__其实不是实例化一个类的时候第一个被调用 的方法,最先被调用的方法 其实是 __new__ 方法。
__new__方法接受的参数虽然也是和__init__一样,但__init__是在类实例创建之后调用,而 __new__方法正是创建这个类实例的方法。
__init__ 和 __new__ 最主要的区别在于:
1.__init__ 通常用于初始化一个新实例,控制这个初始化的过程,比如添加一些属性, 做一些额外的操作,发生在类实例被创建完以后。它是实例级别的方法。
2.__new__ 通常用于控制生成一个新实例的过程。它是类级别的方法。
参考黄哥
__new__和__init__到底是怎么一回事,看下面的代码
如果类没有定义__new__方法,就从父类继承这个__new__方法。
__new__先于__init__执行,类带括号调用时,发生这样的一件事,
先调用类的__new__方法,放回该类的实例对象,这个实例对象就是__init__方法的第一个参数。
请看代码中tmp,self,p的内存地址都是一样的,都是类的实例对象。
#! /usr/bin/python
# coding:utf-8
class Foo(object):
def __new__(cls, *args, **kwargs):
"""
如果不覆盖(override(重写))这个__new__方法,也就是说不写这个__new__方法,类会从object
继承__new__方法完成返回值实例对象
"""
print("__new__方法先被调用")
tmp = super(Foo, cls).__new__(cls, *args, **kwargs)
#(通常情况下是使用 super(Persion, cls).__new__(cls, … …) 这样的方式)
print(id(tmp))
print(type(tmp))
print(isinstance(tmp, Foo))
#isinstance() 函数来判断一个对象是否是一个已知的类型,类似 type()。
#isinstance() 与 type() 区别:
#type() 不会认为子类是一种父类类型,不考虑继承关系。
#isinstance() 会认为子类是一种父类类型,考虑继承关系。
#如果要判断两个类型是否相同推荐使用 isinstance()。
#isinstance(object, classinfo)
#参数
#object -- 实例对象。
#classinfo -- 可以是直接或间接类名、基本类型或者由它们组成的元组。
#返回值
#如果对象的类型与参数二的类型(classinfo)相同则返回 True,否则返回 False。。
print(issubclass(type(tmp), Foo))
#issubclass() 方法用于判断参数 class 是否是类型参数 classinfo 的子类
#issubclass(class, classinfo)
#参数
#class -- 类。
#classinfo -- 类。
#返回值
#如果 class 是 classinfo 的子类返回 True,否则返回 False。
return(tmp)
def __init__(self):
"""self是python默认传的值,该值是调用__new__的返回值"""
print("__init__被调用")
print(id(self))
p = Foo()
print(id(p))
print(type(p))
代码运行结果
__new__方法先被调用
4351203424
True
True
__init__被调用
4351203424
4351203424