Python设计模式之单例模式实例
注:使用的是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__': 输出的前两个结果是相同的(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__': 运行结果是: 复制代码 代码如下: 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,**kwargs): if __name__ == '__main__': 错误如下: 复制代码 代码如下: 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__': 该怎么做呢? 下面参考了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,**kwargs): print 'hhhhhhhhh' if not cls.__instance: cls.__instance = super(Foo,cls).__new__(cls,**kwargs) return cls.__instance def hi(self): if __name__ == '__main__': 运行结果: 复制代码 代码如下: 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,**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,**kwargs): print 'hi,i am Fo' return super(Fo,**kwargs) class Foo(Fo): if __name__ == '__main__': 运行结果如下: 复制代码 代码如下: True True True hi,i am Fo hi,world True True True 如果如下定义Fo,也正常运行: 复制代码 代码如下: class Fo(object): pass 但是,若这样定义: 复制代码 代码如下: class Fo(object): def __new__(cls,i am Fo' 运行时报错如下: 复制代码 代码如下: AttributeError: 'NoneType' object has no attribute 'hi' (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
- python – 熊猫:在最大距离内找到点
- python实现简单图书管理系统
- 在python中从long转换为wide
- mercurial – 为什么TortoiseHg会在提交时随机大写我的文件
- Python获取SQLite查询结果表列名的方法
- 为比特串联,为什么列表比字符数组更快
- Python实现对excel文件列表值进行统计的方法
- Coursea Hadoop课堂笔记——Lesson 1: Big Data Hadoop Sta
- python-3.x – py2app TypeError:dyld_find()得到一个意外
- OpenCV python:cv2.split()vs在BGR图像中获取通道时切片