実現(推奨)法に基づく__new__
我々はオブジェクトをインスタンス化すると、__new__方法の最初の実装は、(我々は、デフォルトの呼び出しオブジェクト.__ new__を書いていない)**クラス**であるオブジェクトをインスタンス化してから再度実行して** __init__メソッドクラス。 **、このオブジェクトが初期化され、すべてこれに基づいて、我々はシングルトンを達成することができます
# 重写__new__方法,实现每一次实例化的时候,返回同一个instance对象
class Singleton:
def __new__(cls, *args, **kwargs):
#判断类cls是否有_instance属性
if not hasattr(cls, '_instance'): # _instance不能改成其他变量,改了单例模式就失效了
#将一个类的实例绑定到类属性_instance上
cls._instance = object.__new__(Singleton)
return cls._instance # 下一个实例化对象返回的是上一个的对象
def __init__(self, name, age):
# 第二个实例对象s2虽然返回的是第一个对象,但是还是会走init方法,把name和age修改了
self.name = name
self.age = age
s1 = Singleton('张三', 24)
s2 = Singleton('李四', 20)
print(s1, s2) # 这两实例都一样
print(s1.name, s2.name)
print(id(s1))
print(id(s2))
注:** __ new__方法は、トリガ__init__メソッドを避けることができない、変数の初期メンバーが上書きされます。**
シングルトンモジュール
Pythonのモジュールは、第1のモジュールに導入され、生成し、天然のシングルモードの実施形態である.pyc
第二の時間を導入する際に、ファイルが直接ロードされ.pyc
なく再びブロック・コードを実行するよりも、ファイル。
#foo1.py
class Singleton(object):
def foo(self):
pass
singleton = Singleton()
#foo.py
from foo1 import singleton
デコレーター
原則:クラスを制御するために使用するデコレータは__call__メソッドを呼び出します。
def singleton(cls, *args, **kw):
instance = {}
def _singleton(args):
if cls not in instance:
instance[cls] = cls(*args, **kw)
return instance[cls]
return _singleton
@singleton
class A:
pass
元のクラスメソッド
'''
此方法是在__new__方法的更上层对实例化过程进行控制。
原理:执行元类的__new__方法和__init__方法用来实例化类对象,__call__ 方法用来对实例化的对象的实例即类的对象进行控制。__call__方法会调用实例类的 __new__方法,用于创建对象。返回对象给__call__方法,然后调用类对象的 __init__方法,用于对对象初始化。
'''
class Singleton1(type):
def __init__(self, *args, **kwargs):
self.__instance = None
super(Singleton1,self).__init__(*args, **kwargs)
def __call__(self, *args, **kwargs):
if self.__instance is None:
self.__instance = super(Singleton1,self).__call__(*args, **kwargs)
return self.__instance
class Singleton2(type):
_inst = {}
def __call__(cls, *args, **kwargs):
print(cls)
if cls not in cls._inst:
cls._inst[cls] = super(Singleton2, cls).__call__(*args)
return cls._inst[cls]
class C(metaclass=Singleton1):
pass
単一モジュールの例外に加えて、いくつかの他のモードの性質は、クラスがインスタンス化されているかどうかを決定するために、中間変数によって提供されます。違いは、中間変数の位置で、またはメタクラスを設定、または関数にカプセル化され、クラス、または静的変数として提供します。
注意:中間変数と変化するプレゼンススレッドの安全性の問題へのアクセス:マルチスレッドモードロック処理を開く必要があります。