ホームページ >バックエンド開発 >Python チュートリアル >Python デザイン パターン シングルトン パターンの例

Python デザイン パターン シングルトン パターンの例

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBオリジナル
2016-06-16 08:44:171148ブラウズ

注:使用的是Python 2.7。

一个简单实现

复制代码 代码如下:

class Foo(object):
    __instance = None
    def __init__(self):
        pass
    @classmethod
    def getinstance(cls):
        if(cls.__instance == None):
            cls.__instance = Foo()
        return cls.__instance

if __name__ == '__main__':
    foo1 = Foo.getinstance()
    foo2 = Foo.getinstance()
    print id(foo1)
    print id(foo2)
    print id(Foo())


输出的前两个结果是相同的(id(foo1)与id(foo2)的值相同),第三个结果和前两个不同。这里类方法getinstance()用于获取单例,但是类本身也可以实例化,这样的方式其实并不符合单例模式的要求。但是这样做也有好处,代码简单,大家约定好这样子调用就行了。但是最好在类的命名上也体现了出来这是一个单例类,例如Foo_singleton。

换一个思路

先说一下init和new的区别:

复制代码 代码如下:

class Foo(object):
    __instance = None
    def __init__(self):
        print 'init'
if __name__ == '__main__':
    foo = Foo()
运行结果是:
复制代码 代码如下:

init
而下面的示例:
复制代码 代码如下:

class Foo(object):
    __instance = None
    def __init__(self):
        print 'init'
    def __new__(cls, *args, **kwargs):
        print 'new'

if __name__ == '__main__':
    foo = Foo()


运行结果是:
复制代码 代码如下:
new

new是一个类方法,会创建对象时调用。而init方法是在创建完对象后调用,对当前对象的实例做一些一些初始化,无返回值。如果重写了new而在new里面没有调用init或者没有返回实例,那么init将不起作用。以下内容引用自http://docs.python.org/2/reference/datamodel.html#object.new

复制代码 代码如下:

If __new__() returns an instance of cls, then the new instance's __init__() method will be invoked like __init__(self[, ...]), where self is the new instance and the remaining arguments are the same as were passed to __new__().

If __new__() does not return an instance of cls, then the new instance's __init__() method will not be invoked.


这样做:
复制代码 代码如下:

class Foo(object):
    __instance = None
    def __init__(self):
        print 'init'

    def __new__(cls, *args, **kwargs):
        print 'new'
        if cls.__instance == None:
            cls.__instance = cls.__new__(cls, *args, **kwargs)
        return cls.__instance

if __name__ == '__main__':
    foo = Foo()

    错误如下:

复制代码 代码如下:

RuntimeError: maximum recursion depth exceeded in cmp

而这样也有一样的错误:

复制代码 代码如下:

class Foo(object):
    __instance = None
    def __init__(self):
        if self.__class__.__instance == None:
            self.__class__.__instance = Foo()
        print 'init'

if __name__ == '__main__':
    foo = Foo()


该怎么做呢?

以下は http://stackoverflow.com/questions/31875/is-there-a-simple-elegant-way-to-define-singletons-in-python/31887#31887 を参照しています:

コードをコピー コードは次のとおりです:

class Foo(object):
__instance = None
def __new__(cls, *args, * *KWARGS):
Print 'うーん、 )。CLS .__インスタンス

def hi(self):
print 'hi, world' print 'hi, letian'


if __name__ == '__main__':
foo1 = Foo()

foo2 = Foo()

print id(foo1)
print id(foo2)
print isinstance(foo1) , object)
print isinstance(foo1, Foo)
foo1.hi()

実行結果:



コードをコピー
コードは次のとおりです。ああああああああああああ
39578896
39578896
True
True
こんにちは、世界
こんにちは、letian

それで、何が起こったのか、まず超おさらいしましょう:

コードをコピー

コードは次のとおりです。>>> print super.__doc__super( type) - > バインドされていないスーパー オブジェクト
super(type, obj) -> バインドされたスーパー オブジェクト; isinstance(obj, type)
super(type, type2) -> バインドされたスーパー オブジェクトが必要type2, type )
協調スーパークラス メソッドを呼び出すための一般的な使用法:
class C(B):
def meth(self, arg):
super(C, self).meth(arg)

上記のシングルトン パターン コードのこのコード行を確認してください:


コードをコピーします
コードは次のとおりです:cls .__instance = super(Foo, cls).__new__(cls, *args, **kwargs)
super(Foo, cls) はオブジェクトです、super(Foo, cls) ).new メソッドはオブジェクトの new メソッドを使用します。 object.new メソッドの機能を見てみましょう:


コードをコピー
コードは次のとおりです: > ;>> print object .__new__.__doc__T.__new__(S, ...) -> T

のサブタイプを持つ新しいオブジェクト
継承チェーンの場合


コードをコピーします

コードは次のとおりです:class Fo(object ): def __new__(cls, *args, **kwargs):
print 'こんにちは、私は Fo'
return super(Fo, cls).__new__(cls, * args、**kwargs )

class Foo(Fo):
__instance = None def __new__(cls, *args, **kwargs):

if not cls.__instance:
print Foo は cls
print issubclass(cls, Fo)
print issubclass(cls, object)
cls.__instance = super(Foo, cls).__new__(cls, *args, **kwargs)
return cls.__instance

def hi(self):
print 'hi, world'

if __name__ == '__main__':
foo1 = Foo()

foo1.hi()

print isinstance(foo1, Foo)
print isinstance(foo1, Fo)
print isinstance(foo1, object)

実行結果は次のとおりです:



コードをコピー
コードは次のとおりです:真 真

こんにちは、私は Fo
こんにちは、世界




Fo が定義されている場合


コードをコピー
コードは次のとおりです:class Fo(object): pass

ただし、次のように定義した場合:


コードをコピー
コードは次のとおりです:class Fo( object): def __new__(cls, * ARGS, ** KWARGS):
Print 'Hi, I am fo'

実行時レポートは次のように間違っています:


コードをコピーします
コードは次のようになります:

AttributeError: 'NoneType' オブジェクトには属性 'hi' がありません
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。