设为首页 收藏本站
查看: 954|回复: 0

[经验分享] Python单例模式和Borg惯用法及相关问题

[复制链接]

尚未签到

发表于 2017-5-6 10:07:38 | 显示全部楼层 |阅读模式
  一、单例模式
  如果你想保证某个类从始至终最多只能有一个实例,那么单例模式可能会是你首先想到的,使用__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)
  执行后,发现在子类中加入[size=1em]_state = {} 就可以实现:自己的类实例能够相互共享状态,但却不和Borg的其它子类共享:

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

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-373672-1-1.html 上篇帖子: python:将数字转换成用英文表达的程序 下篇帖子: Python基础教程之第6章 抽象
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表