单例设计模式:
目的:让类创建的对象,在系统中只有唯一的一个实例且每一次执行类名返回的对象,内存地址是相同的。
(1)__new__方法:
1、为对象分配空间
2、返回对象引用
而__init__方法:
1、对象初始化
2、定义实例属性
在使用类名创建对象是,python的解释器首先会调用__new__方法为对象分配空间,它是一个由object基类提供的内置的静态方法。
python的解释器获得对象的引用后,将引用作为第一个参数,传递给——init——方法。
(2)重写——new——方法
重写new方法的代码非常固定
重写new方法一定要return super().new(cls),否则python解释器得不到分配了空间的对象引用,就不会调用对象的初始化方法
注意:new方法是一个静态方法,在调用是需要主动传递cls参数。
例子:先做一个简单的初始化方法
class Musicplayer(): def __init__(self): print("播放器初始化") ##创建类的对象 player=Musicplayer() print(player)
输出结果为播放器初始化,同时也把对象的内存地址也做了一个输出。
播放器初始化 <__main__.Musicplayer object at 0x0000026E241C65F8>
然后重写object的new方法
class Musicplayer(): #定义一个类属性,记录第一个被创建对象的使用 instance=None def __new__(cls, *args, **kwargs): ##1、判断类属性是否是空对象 if cls.instance==None: ## 2、调用父类方法,为第一个对象分配空间 cls.instance=super().__new__(cls) ##3、返回类属性的单例引用 return cls.instance def __init__(self): print("播放器初始化") ##创建类的对象 player1=Musicplayer() print(player1) player2=Musicplayer() print(player2)
输出结果 播放器初始化 <__main__.Musicplayer object at 0x000002260F05B630> 播放器初始化 <__main__.Musicplayer object at 0x000002260F05B630>
从以上结果看出,地址是一样的了,但是初始化方法中被调用了两次。
有没有什么解决办法让初始化动作只被执行一次的呐。
步骤:
1、定义一个类属性init_flag标记是否执行过初始化动作,初始值为False,
2、在__init__方法中,判断init_flag,如果 为False,就执行初始化动作
3、然后将init_flag设置为True
class Musicplayer(): #定义一个类属性,记录第一个被创建对象的使用 instance=None #定义一个类属性,标记是否执行过初始化动作 init_flag=False def __new__(cls, *args, **kwargs): ##1、判断类属性是否是空对象(判断类属性采用类名.的方式去查找类属性) if cls.instance==None: ## 2、调用父类方法,为第一个对象分配空间 cls.instance=super().__new__(cls) ##3、返回类属性保存的对象引用 return cls.instance def __init__(self): if Musicplayer.init_flag:##因为这是类属性,所以采用类名.的方式来查找类属性 return##如果为真就直接返回,否则执行初始化动作 print("播放器初始化") Musicplayer.init_flag=True ##创建类的对象 player1=Musicplayer() print(player1) player2=Musicplayer() print(player2)
结果如下:
播放器初始化 <__main__.Musicplayer object at 0x0000012BA2E6C630> <__main__.Musicplayer object at 0x0000012BA2E6C630>