在Python中,__new__
方法是一个魔法方法,用于在创建新实例之前控制对象的创建。它是构造器方法__init__
之前被调用的。__new__
方法通常用于控制实例的创建,可以返回实例本身或者一个完全不同的对象。
__new__ 方法
使用 类名() 创建对象时,Python 的解释器 首先 会 调用 __new__ 方法为对象 分配空间
__new__ 是一个 由 object 基类提供的 内置的静态方法,主要作用有两个:
在内存中为对象 分配空间
返回 对象的引用
Python 的解释器获得对象的 引用 后,将引用作为 第一个参数,传递给 __init__ 方法
- 重写
__new__
方法 一定要return super().__new__(cls)
- 否则 Python 的解释器 得不到 分配了空间的 对象引用,就不会调用对象的初始化方法
- 注意:
__new__
是一个静态方法,在调用时需要 主动传递cls
参数
需求
让 初始化动作 只被 执行一次
解决办法
定义一个类属性 init_flag 标记是否 执行过初始化动作,初始值为 False
在 __init__ 方法中,判断 init_flag,如果为 False 就执行初始化动作
然后将 init_flag 设置为 True
这样,再次 自动 调用 __init__ 方法时,初始化动作就不会被再次执行 了
class MusicPlayer(object): instance = None # 记录是否执行过初始化动作 init_flag = False def __new__(cls, *args, **kwargs): cls.key = 'kkk' # 1. 判断类属性是否已经被赋值 if cls.instance is None: cls.instance = super().__new__(cls) return cls.instance def __init__(self, value): if not MusicPlayer.init_flag: self.value = value MusicPlayer.init_flag = True def show(self): print(self.key) print(self.value) if __name__ == "__main__": # 创建第一个实例 p1 = MusicPlayer(10) p1.show() # kkk 10 p2 = MusicPlayer(20) p2.show() # kkk 10
__init__为初始化方法,__new__方法是真正的构造函数。
__new__是实例创建之前被调用,它的任务是创建并返回该实例,是静态方法
__init__是实例创建之后被调用的,然后设置对象属性的一些初始值。
总结:__new__方法在__init__方法之前被调用,并且__new__方法的返回值将传递给__init__方法作为第一个参数,最后__init__给这个实例设置一些参数。