python类继承重复_python单例模式,可以继承,不会重复执行初始化函数的版本
網(wǎng)上最長(zhǎng)見(jiàn)的版本:
1 classSingleton(object):2 __instance=None3 def__init__(self):4 pass
5 def__new__(cls,*args,**kwd):6 if Singleton.__instance isNone:7 Singleton.__instance=object.__new__(cls,*args,**kwd)8 return Singleton.__instance
坑1:每次實(shí)例化時(shí),__init__()都會(huì)執(zhí)行一次
原因:python每次實(shí)力化一個(gè)類(lèi)先調(diào)用__new__方法,再調(diào)用__init__方法。在__new__方法中,在不讓子類(lèi)有感知的情況下,無(wú)法跳過(guò)該機(jī)制
坑2:該類(lèi)無(wú)法繼承,如果有多個(gè)類(lèi)繼承了該基類(lèi),那么每次實(shí)例化不同的類(lèi)得到的結(jié)果都是第一次實(shí)例化時(shí)得到的類(lèi)
原因:__instance作為Singleton這個(gè)類(lèi)的類(lèi)變量存在,當(dāng)?shù)谝淮螌?shí)例化該類(lèi)的子類(lèi)時(shí),__instance被設(shè)置為實(shí)例化的結(jié)果,之后實(shí)例化每個(gè)子類(lèi)時(shí),在__new__函數(shù)中檢測(cè)到該變量已設(shè)置就不會(huì)再去實(shí)例化,所以該值也就不會(huì)變化。
改進(jìn):
classSingleton(object):def __new__(cls, *args, **kwargs):if not hasattr(cls, '__instance'):print('in new')
cls.__instance = object.__new__(cls, *args, **kwargs)
cls.__instance.__Singleton_Init__(*args, **kwargs)return cls.__instance
def __Singleton_Init__(self):print("__Singleton_Init__")
填坑1:該類(lèi)規(guī)定子類(lèi)的初始化函數(shù)是__Singleton_Init__,這樣就可以在__new__方法中控制初始化方法的調(diào)用
填坑2:__new__方法的cls參數(shù)是最上的層子類(lèi),判斷cls.__instance是否被設(shè)置就是在判斷各個(gè)子類(lèi)是否有__instance成員變量,這樣就可以愉快的繼承了
再挖個(gè)坑:if not hasattr(cls, '__instance'),為什么這里可以檢查cls的“私有成員”
一個(gè)例子:
#-*- coding: utf-8 -*-
classSingleton(object):def __new__(cls, *args, **kwargs):if not hasattr(cls, '__instance'):print('before new')print(cls)
cls.__instance = object.__new__(cls, *args, **kwargs)print('after new')
cls.__instance.__Singleton_Init__(*args, **kwargs)return cls.__instance
def __init__(self):print("__init__")def __Singleton_Init__(self):print("__Singleton_Init__")classBB(Singleton):pass
classCC(Singleton):passc=CC()
c1=CC()
b=BB()
b.a=2c.a=3
print(id(c), id(c1))print(b.a, c.a)
總結(jié)
以上是生活随笔為你收集整理的python类继承重复_python单例模式,可以继承,不会重复执行初始化函数的版本的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: python实验三答案_20194123
- 下一篇: python里orient_从Pytho