Python设计模式之单例模式实例


Posted in Python onApril 26, 2014

注:使用的是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 'hhhhhhhhh'
        if not cls.__instance:
            cls.__instance = super(Foo, cls).__new__(cls, *args, **kwargs)
        return cls.__instance
    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()

运行结果:
hhhhhhhhh
hhhhhhhhh
39578896
39578896
True
True
hi, world
hi, letian

那么,到底发生了什么,我们先回顾一下super:
>>> print super.__doc__
super(type) -> unbound super object
super(type, obj) -> bound super object; requires isinstance(obj, type)
super(type, type2) -> bound super object; requires issubclass(type2, type)
Typical use to call a cooperative superclass method:
class C(B):
    def meth(self, arg):
        super(C, self).meth(arg)

可以肯定上面的单例模式代码中的这一行代码:
cls.__instance = super(Foo, cls).__new__(cls, *args, **kwargs)

super(Foo, cls)是object,super(Foo, cls).new方法使用的是object的new方法。我们看一下object.new方法的作用:
>>> print object.__new__.__doc__
T.__new__(S, ...) -> a new object with type S, a subtype of T

如果是一个继承链

class Fo(object):
    def __new__(cls, *args, **kwargs):
        print 'hi, i am 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 is 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)

运行结果如下:
True
True
True
hi, i am Fo
hi, world
True
True
True

如果如下定义Fo,也正常运行:
class Fo(object):
    pass

但是,若这样定义:
class Fo(object):
    def __new__(cls, *args, **kwargs):
        print 'hi, i am Fo'

运行时报错如下:
AttributeError: 'NoneType' object has no attribute 'hi'
Python 相关文章推荐
python 排序算法总结及实例详解
Sep 28 Python
python算法演练_One Rule 算法(详解)
May 17 Python
基于Python socket的端口扫描程序实例代码
Feb 09 Python
Python实现对文件进行单词划分并去重排序操作示例
Jul 10 Python
对Python实现累加函数的方法详解
Jan 23 Python
Python求离散序列导数的示例
Jul 10 Python
python数据处理之如何选取csv文件中某几行的数据
Sep 02 Python
python读取word 中指定位置的表格及表格数据
Oct 23 Python
python sklearn包——混淆矩阵、分类报告等自动生成方式
Feb 28 Python
Django获取model中的字段名和字段的verbose_name方式
May 19 Python
在keras中实现查看其训练loss值
Jun 16 Python
python 第三方库paramiko的常用方式
Feb 20 Python
Python设计模式之观察者模式实例
Apr 26 #Python
Python设计模式之代理模式实例
Apr 26 #Python
python中的列表推导浅析
Apr 26 #Python
Python中的Numpy入门教程
Apr 26 #Python
Python中的map、reduce和filter浅析
Apr 26 #Python
Python实现的Kmeans++算法实例
Apr 26 #Python
爬山算法简介和Python实现实例
Apr 26 #Python
You might like
解决form中action属性后面?传递参数 获取不到的问题
2017/07/21 PHP
php探针不显示内存解决方法
2019/09/17 PHP
PHP之多条件混合筛选功能的实现方法
2019/10/09 PHP
js实现获取div坐标的方法
2015/11/16 Javascript
详解AngularJS Filter(过滤器)用法
2015/12/28 Javascript
js实现楼层效果的简单实例
2016/07/15 Javascript
AngularJS 工作原理详解
2016/08/18 Javascript
AngularJS入门教程之双向绑定详解
2016/08/18 Javascript
Ajax与服务器(JSON)通信实例代码
2016/11/05 Javascript
微信小程序 数据交互与渲染实例详解
2017/01/21 Javascript
微信小程序  TLS 版本必须大于等于1.2问题解决
2017/02/22 Javascript
Express之get,pos请求参数的获取
2017/05/02 Javascript
vue的状态管理模式vuex
2017/11/30 Javascript
vue-cli2.9.3 详细教程
2018/04/23 Javascript
在小程序中使用腾讯视频插件播放教程视频的方法
2018/07/10 Javascript
模块化react-router配置方法详解
2019/06/03 Javascript
python实现调用其他python脚本的方法
2014/10/05 Python
Python实现单词拼写检查
2015/04/25 Python
python编写分类决策树的代码
2017/12/21 Python
详解用python实现基本的学生管理系统(文件存储版)(python3)
2019/04/25 Python
Python数据可视化 pyecharts实现各种统计图表过程详解
2019/08/15 Python
python定位xpath 节点位置的方法
2019/08/27 Python
Python lxml模块的基本使用方法分析
2019/12/21 Python
pycharm无法导入本地模块的解决方式
2020/02/12 Python
Keras构建神经网络踩坑(解决model.predict预测值全为0.0的问题)
2020/07/07 Python
python获取命令行参数实例方法讲解
2020/11/02 Python
Lenox官网:精美的瓷器&独特的礼品
2017/02/12 全球购物
美国床垫和床上用品公司:Nest Bedding
2017/06/12 全球购物
TheFork葡萄牙:欧洲领先的在线餐厅预订平台
2019/05/27 全球购物
简述DNS进行域名解析的过程
2013/12/02 面试题
学习新党章思想汇报
2014/01/09 职场文书
医学生个人求职信范文
2014/02/07 职场文书
中餐厅经理岗位职责
2014/04/11 职场文书
最美乡村医生事迹材料
2014/06/02 职场文书
市委常委会班子党的群众路线教育实践活动整改方案
2014/10/25 职场文书
python 解决微分方程的操作(数值解法)
2021/05/26 Python