类的单例模式
单例模式(Singleton)也叫单态模式,是设计模式中最为简单的一种模式,甚至有些模式大师都不称其为模式,称其为一种实现技巧,因为设计模式讲究对象之间的关系的抽象,而单例模式只有自己一个对象,也因此有些设计大师并把把其称为设计模式之一。
这里又不具体讲如何实现单例模式和介绍其原理(因为这方便的已经有太多的好文章介绍了),如果对单例模式不了解的可以先看下:http://terrylee.cnblogs.com/archive/2005/12/09/293509.html 。当然也可以自己搜索。
好多没怎么使用过的人可能会想,单例模式感觉不怎么用到,实际的应用场景有哪些呢?以下,我将列出一些就在咱们周边和很有意义的单例应用场景。
1. Windows的Task Manager(任务管理器)就是很典型的单例模式(这个很熟悉吧),想想看,是不是呢,你能打开两个windows task manager吗? 不信你自己试试看哦~
2. windows的Recycle Bin(回收站)也是典型的单例应用。在整个系统运行过程中,回收站一直维护着仅有的一个实例。
3. 网站的计数器,一般也是采用单例模式实现,否则难以同步。
4. 应用程序的日志应用,一般都何用单例模式实现,这一般是由于共享的日志文件一直处于打开状态,因为只能有一个实例去操作,否则内容不好追加。
5. Web应用的配置对象的读取,一般也应用单例模式,这个是由于配置文件是共享的资源。
6. 数据库连接池的设计一般也是采用单例模式,因为数据库连接是一种数据库资源。数据库软件系统中使用数据库连接池,主要是节省打开或者关闭数据库连接所引起的效率损耗,这种效率上的损耗还是非常昂贵的,因为何用单例模式来维护,就可以大大降低这种损耗。
7. 多线程的线程池的设计一般也是采用单例模式,这是由于线程池要方便对池中的线程进行控制。
8. 操作系统的文件系统,也是大的单例模式实现的具体例子,一个操作系统只能有一个文件系统。
9. HttpApplication 也是单位例的典型应用。熟悉ASP.Net(IIS)的整个请求生命周期的人应该知道HttpApplication也是单例模式,所有的HttpModule都共享一个HttpApplication实例。
总结以上,不难看出:
单例模式应用的场景一般发现在以下条件下:
(1)资源共享的情况下,避免由于资源操作时导致的性能或损耗等。如上述中的日志文件,应用配置。
(2)控制资源的情况下,方便资源之间的互相通信。如线程池等。
一、实例化类
实例化一个类时 1. 创建一个对象,调用__new__方法,如果没有会调用父类的__new__方法 2. 调用__init__方法 3. 返回对象的引用
class Dog(object): def __init__(self): print("---init方法---") def __new__(cls, *args, **kwargs): print("---new方法---") xtq = Dog() #---------结果--------- ---new方法---
二、单例
class Dog(object): _instance = None _init_flag = False def __new__(cls, *args , **kwargs): #cls指向Dog if cls._instance == None: cls._instance = object.__new__(cls) return cls._instance else: return cls._instance def __init__(self,name): if Dog._init_flag == False: self.name = name Dog._init_flag = True a = Dog("旺财") print(a.name) b = Dog("啸天犬") print(b.name) #----------------结果----------------- 旺财 旺财
创建单例模式:有三种方式
方式一:文件导入的方式,可参考CRM项目的 方式二:类方法或者静态方法 class Foo: _instance = None def __init__(self): pass @classmethod def get_instance(cls): if cls._instance: return cls._instance else: obj = cls() cls._instance = obj return obj a1 = Foo.get_instance() a2 = Foo.get_instance() a3 = Foo.get_instance() print(a1,a2,a3,a4) 输出: <__main__.Foo object at 0x0000000002950C18> <__main__.Foo object at 0x0000000002950C18> <__main__.Foo object at 0x0000000002950C18> 方式三:__new__方法 class Foo: _instance = None def __init__(self): pass def __new__(cls, *args, **kwargs): if cls._instance: return cls._instance else: obj = object.__new__(cls, *args, **kwargs) cls._instance = obj return obj obj1 = Foo() obj2 = Foo() obj3 = Foo() print(obj1, obj2, obj3) 有何用: - 自定义CURD组件时 - 发布文章,特殊字符过滤 KindEditor class Kind: def __init__(self): self.valid_tags = [ "a","div","H1" ] def valid(self,content): .... obj = Kind() body = obj.valid("<html>....")