`
shuiyutian
  • 浏览: 10565 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Python单例模式和Borg惯用法及相关问题

阅读更多

    一、单例模式

    如果你想保证某个类从始至终最多只能有一个实例,那么单例模式可能会是你首先想到的,使用__new__静态方法可以很简单的解决:

class Singleton(object):
    def __new__(cls,*args,**kwargs):
        if '_inst' not in vars(cls):
            cls._inst = super(Singleton,cls).__new__(cls)
        return cls._inst

    Python 2.X 里你可能看到的实现代码:

class Singleton(object):
    def __new__(cls,*args,**kwargs):
        if '_inst' not in vars(cls):
            cls._inst = super(Singleton,cls).__new__(cls,*args,*kwargs)
        return cls._inst

    它们这间的细微不同之处在于 *args,**kwargs:

cls._inst = super(Singleton,cls).__new__(cls,*args,**kwargs)

    而这在Python 3.x里将会引发异常:TypeError: object() takes no parameters

    如果我们查看 CPython 源码,我们将会看到:

    (1)在__new__ 被overridden或者__init__没有被overridden 的情况下,如果调用 object.__new__的时候传递了除cls之外的参数将会报错;

    (2)__new__ 没有被overridden或者__init__被overridden 的情况下,如果调用 object.__init__  的时候传递了除cls之外的参数将会报错;

    (3)*args,和*kwds 在object.__new__除了用来判断报错,并没有什么其它用处

 

static int
object_init(PyObject *self, PyObject *args, PyObject *kwds)
{
    int err = 0;
    PyTypeObject *type = Py_TYPE(self);
    if (excess_args(args, kwds) &&
        (type->tp_new == object_new || type->tp_init != object_init)) {
        PyErr_SetString(PyExc_TypeError, "object.__init__() takes no parameters");
        err = -1;
    }
    return err;
}

static PyObject *
object_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
{
    if (excess_args(args, kwds) &&
        (type->tp_init == object_init || type->tp_new != object_new)) {
        PyErr_SetString(PyExc_TypeError, "object() takes no parameters");
        return NULL;
    }
    ...
}

    

    二、Borg惯用法

    如果你想保证某个类从始至终只创建了一个实例:你并不关心生成实例的id,只关心其状态和行为方式,而且你还想确保它具有子类化的能力,那么可以使用Borg惯用法:

    

class Borg(object):
    _state = {}
    def __new__(cls,*args,**kwargs):
        obj = super(SingletonBorg,cls).__new__(cls)
        obj.__dict__ = cls._state
        return obj

    和单例模式不同的是,Borg惯用法允许多个实例被创建,但所有的实例都共享状态和行为方式。通过这种“数据重载”,你的类不会从Borg继承_shared_state属性,而是自己的数据。为了允许这种“数据重载”,Borg的__new__应当使用cls._shared_state,而不是Borg._shared_state。

    如果你想Borg的所有类实例都能够共享状态:

class SingletonSpam_02(Borg):pass

    如果你想你的类实例能够相互共享状态,但却不和Borg的其它子类共享,需要在类作用域中加入这样的声明:

     class SingletonSpam_03(Borg):
         _state = {}

    测试:

 if __name__ == '__main__':   
    class SingletonSpam_02(Borg):pass
    class SingletonSpam_03(Borg):
         _state = {}
    one = SingletonSpam_02()
    two = SingletonSpam_02()
    one.name = 'lilei'
    two.name = 'hanmeimei'
    print(one.name,two.name)
 
    three = SingletonSpam_03()
    four  = SingletonSpam_03()
    three.name = 'liuxing'
    four.name  = 'baiyun'
    print(three.name,four.name)

    执行后,发现在子类中加入_state = {} 就可以实现:自己的类实例能够相互共享状态,但却不和Borg的其它子类共享

('hanmeimei', 'hanmeimei')
('baiyun', 'baiyun')

 

    

分享到:
评论

相关推荐

    聊聊python里如何用Borg pattern实现的单例模式

    主要介绍了聊聊python里如何用Borg pattern实现的单例模式,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

    Python库 | borg_space-0.0.0-py3-none-any.whl

    资源分类:Python库 所属语言:Python 资源全名:borg_space-0.0.0-py3-none-any.whl 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059

    Python Cookbook

    6.16 用Borg惯用法来避免“单例”模式 259 6.17 Null对象设计模式的实现 263 6.18 用_ _init_ _参数自动初始化实例变量 266 6.19 调用超类的_ _init_ _方法 267 6.20 精确和安全地使用协作的超类调用 270 第7...

    python-patterns-收集了 Python 常用的设计模式-python

    当前模式创建模式:模式描述abstract_factory使用具有特定工厂的通用函数borg一个在实例之间共享状态的单例builder而不是使用多个构造函数,builder对象接收参数并返回构造的对象工厂委托一个专门的函数/方法来创建...

    Python单体模式的几种常见实现方法详解

    本文实例讲述了Python单体模式的几种常见实现方法。分享给大家供大家参考,具体如下: 这里python实现的单体模式,参考了:...

    随机信号处理经典谱与参数谱法matlab实现功率谱估计.pdf

    本文主要介绍对比了两种经典谱估计方法:周期图法与自相关...包含周期图法、自相关法以及三种参数谱估计法:直接求解yule-walker方程,Levinson-Durbin快速递推法与Borg算法的matlab和python代码。以及大量实验结果图。

    Borg9

    Borg9

    Python库 | borg_pod-1.0.1.tar.gz

    资源分类:Python库 所属语言:Python 资源全名:borg_pod-1.0.1.tar.gz 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059

    开源项目-crufter-borg.zip

    开源项目-crufter-borg.zip,Borg - A terminal based search engine for bash commands, built with Go

    BORG :一个快速进化的僵尸网络.pdf

    BORG :一个快速进化的僵尸网络 数据安全 工控安全 金融安全 安全威胁 企业安全

    Python Cookbook英文版

    Python Cookbook英文版 Table of Contents Foreword Preface 1. Python Shortcuts 1.1 Swapping Values Without Using a Temporary Variable 1.2 Constructing a Dictionary Without Excessive Quoting 1.3...

    borg-docker-volume:使用Borg备份Docker卷内容的脚本

    对于使用Borg备份Docker卷的Bash脚本,如果它具有除Bash,Docker和Borg之外的任何依赖关系,我会感到惊讶。 要安装Borg,请按照提供的说明进行操作。 要运行此脚本,您需要位于可以执行sudo(通常是wheel )的用户...

    borg2.28源代码(反汇编工具).

    反汇编工具Borg2.28的源代码,静态反汇编可执行文件的工具Borg的源代码,Borg是一个小巧实用的程序,它允许用户可以静态反汇编可执行文件Win32PE并使用调用栈返回值。 此外,Borg使您能够反汇编成代码或数据文件。 ...

    btrfs-borg:btrfs-borg制作快照并使用Borg备份btrfs快照子卷和LXC容器

    btrfs-borg Btrfs-borg制作快照,并使用Borg备份btrfs快照子卷的列表。 它也可以选择备份目录列表,而无需先为其创建快照。 它包括对由LXC的btrfs后端管理的容器的特殊支持。 最后,它支持构建不会备份的目录清单。...

    Large-scale cluster management at Google with Borg

    Large-scale cluster management at Google with Borg 这是一篇关于混合负载调度的经典论文,主要介绍针对不同负载类型进行混合调度的框架和实现方法。

    Borg-BackUP-GUI:BORG BackUP GUI是用于管理Borg备份的简单GUI。 目前,我正在开发一些扩展

    许多Borg用户确切地知道他们在做什么,只需要该程序即可简单地创建或删除新快照。 新的Borg用户还可以使用BORG BackUP GUI快速地到达目的地。 当然必须安装BorgBackup(简称:Borg) 在这里您将找到有关博格的...

    Python Cookbook, 2nd Edition

    Avoiding the "Singleton" Design Pattern with the Borg Idiom Recipe 6.17. Implementing the Null Object Design Pattern Recipe 6.18. Automatically Initializing Instance Variables from _ _init_ _ ...

    Aglyph:Aglyph是用于Python的依赖注入框架。-开源

    Aglyph可以组装“原型”组件(每次都会创建一个新实例),“单例”组件(每次都会返回同一个实例),“ borg”组件(每次都会创建一个新实例,但是所有相同的实例类共享相同的内部状态)和“弱引用”组件(只要正在...

Global site tag (gtag.js) - Google Analytics