用元类实现单例

lim*_*boy 5 python metaclass python-2.x

下面的代码是我如何使用元类实现单例,并且它运行良好

class Test_MetaClass(type):

    def __init__(cls, name, bases, dict):
        super(Test_MetaClass, cls).__init__(cls, bases, dict)
        cls._instance = None
        print 'Test_MetaClass __init__'

    def __call__(cls, *args, **kwargs):
        if cls._instance is None:
            cls._instance = super(Test_MetaClass, cls).__call__(*args, **kwargs)
        print 'Test_MetaClass __call__'
        return cls._instance


class A(object):
    __metaclass__ = Test_MetaClass
    def __init__(self):
        print 'A __init__ triggered'

a = A()
b = A()
Run Code Online (Sandbox Code Playgroud)

输出:

Test_MetaClass __init__
A __init__ triggered
Test_MetaClass __call__
Test_MetaClass __call__
Run Code Online (Sandbox Code Playgroud)

我的问题是为什么b = A()直接去Test_MetaClass.__call__和忽略__init__

Ery*_*Sun 2

这不是你想要的吗?cls._instance不是 None,所以它不会执行type.__call__(cls, *args, **kwargs)

>>> type.__call__(A)
A __init__ triggered
<__main__.A object at 0x00BADB30>
Run Code Online (Sandbox Code Playgroud)

正是通过这个调用,A.__new__并被A.__init__调用来创建/初始化一个新实例。但对于你的单身人士来说,你只需要一个实例。