python类继承重复_python单例模式,可以继承,不会重复执行初始化函数的版本
網(wǎng)上最長見的版本:
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:每次實例化時,__init__()都會執(zhí)行一次
原因:python每次實力化一個類先調(diào)用__new__方法,再調(diào)用__init__方法。在__new__方法中,在不讓子類有感知的情況下,無法跳過該機(jī)制
坑2:該類無法繼承,如果有多個類繼承了該基類,那么每次實例化不同的類得到的結(jié)果都是第一次實例化時得到的類
原因:__instance作為Singleton這個類的類變量存在,當(dāng)?shù)谝淮螌嵗擃惖淖宇悤r,__instance被設(shè)置為實例化的結(jié)果,之后實例化每個子類時,在__new__函數(shù)中檢測到該變量已設(shè)置就不會再去實例化,所以該值也就不會變化。
改進(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:該類規(guī)定子類的初始化函數(shù)是__Singleton_Init__,這樣就可以在__new__方法中控制初始化方法的調(diào)用
填坑2:__new__方法的cls參數(shù)是最上的層子類,判斷cls.__instance是否被設(shè)置就是在判斷各個子類是否有__instance成員變量,這樣就可以愉快的繼承了
再挖個坑:if not hasattr(cls, '__instance'),為什么這里可以檢查cls的“私有成員”
一個例子:
#-*- 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单例模式,可以继承,不会重复执行初始化函数的版本的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: python实验三答案_20194123
- 下一篇: python里orient_从Pytho