我看到了这个 Borg 单例模式代码,但我无法理解我添加到单例对象的新成员是如何附加到 __shared_state = {} 字典的。

这是单例代码

class Borg(object):
    _shared_state = {}

    def __new__(cls,*args,**kwargs):
        obj = super(Borg,cls).__new__(cls,*args,**kwargs)
        obj.__dict__ = cls._shared_state
        return obj

class Child(Borg):
    pass

if __name__ == '__main__':
    borg = Borg()
    another_borg = Borg()

    print borg is another_borg
    child = Child()

    borg.only_one_var = "I'm the only one var"
    print child.only_one_var

所以我的问题是,当对象 borg.only_one_var 被创建时,它是如何附加到 _shared_state 字典的

最佳答案

默认情况下,每个实例都有自己的字典,因此将属性分配给一个实例不会影响其他实例。

但是你可以让一个实例的字典指向一个新的字典,当你在内部这样做时,它将被用来存储项目。

在您的情况下,每次创建实例时,您都会将其字典分配给 Borg. _shared_state 。因此,它的所有实例都将使用相同的 dict 来获取和设置属性。

它基本上相当于:

shared = {}

class A(object):
    def __init__(self):
        self.__dict__ = shared

演示:
>>> ins = [A() for _ in range(5)]
>>> ins[0].x = 100
>>> for i in ins:
...     print(i.x)
...
100
100
100
100
100

>>> shared
{'x': 100}

在 CPython 中,将新字典分配给 __dict__ 发生在 PyObject_GenericSetDict 中:
int
PyObject_GenericSetDict(PyObject *obj, PyObject *value, void *context)
{
    PyObject **dictptr = _PyObject_GetDictPtr(obj);
    ...
    if (!PyDict_Check(value)) {
        PyErr_Format(PyExc_TypeError,
                     "__dict__ must be set to a dictionary, "
                     "not a '%.200s'", Py_TYPE(value)->tp_name);
        return -1;
    }
    Py_INCREF(value);
    Py_XSETREF(*dictptr, value);  # Set the dict to point to new dict
    return 0;
}

请注意,由于 key-sharing dictionaries 在 Python 3.3+ 中的到来,同一类实例的字典可以共享一些内部状态以节省空间。

关于python - 理解 Python 中的 Borg 单例模式,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/45312657/

10-12 02:54