单例模式
单例模式,即一个类只能有一个实例(这里说对象A),也就是说,不管实例化多少个对象,都共同拥有那一个实例(对象A)的内存,即所有的对象都可以看成是对对象A的引用。
__new__方法必须有一个返回值,且必须返回当前类的一个实例对象,如果没有return语句,则不会执行init方法,所以也就不会对实例初始化
__new__方法必须返回和当前类一致的一个实例,父类也不行,否则不会执行__init__方法。
设计思想
定义一个类属性的对象。给其初始化为0,在new方法去生成实例对象的时候进行判断,如果a==0,就说明这是第一次实例化对象,那么就生成一个实例化对象,并把这个实例化对象赋值给a,如果a不等于0,就说明已经创建过实例化对象了,此时就不需要再重新创建一个实例化对象,直接返回第一次创建的实例化对象即可。
关于类属性和实例属性,我是这样理解的:
实例属性,是每个对象单独拥有的。类属性是大家共同拥有的。
可以类比租房子。假设有3个人合租房子,房子是3个卧室还有一个客厅。那么卧室就是实例属性,只能自己访问。客厅就是类属性,大家共同享有,且每个人都可以对其进行修改。
代码如下
class Bus317(object):
#类属性(内存是公用的)
line="317"
a=0
# __init__方法里面的属性是实例属性(每个对象都有一个实例属性,会给每个实例对象开辟不同的内存)
def __init__(self,car_num,name):
self.car_name=car_num
self.name = name
self.line="318"
self.lst=[]
print("创建对象")
def __new__(cls, *args, **kwargs):
if Bus317.a==0:
Bus317.a = object.__new__(cls) #object.new创建一个实例
return Bus317.a
print("xxxxx") #只会执行一次
def run(self,flag):
print("bus317 run")
if flag==0:
print("从A站开往C站")
else:
print("从C站开往A站")
bus02=Bus317("hk","dsgg")
bus01=Bus317("hsak","asggg")
bus=Bus317("hak","ggg")
print(id(bus02),id(bus01),id(bus))
运行截图:
可以看到,最开始输出的XXXXX,是在定义类的时候,输出的,只会执行一次,因为这是属于类属性的,所以在创建对象的时候不会执行。只是在类创建的时候会执行一次。那么很显然,类属性a是属于类的,所以不论创建多少个实例化对象,在内存里面为对象开辟空间的时候,都不会为类属性a分配空间,只会建立类似于指针的东西,让其指向a的内存(类创建时分配的空间),会为每个实例化对象分配实例属性的内存,因为这是每个对象所独自拥有的。