Python设计模式中单例模式的实现及在Tornado中的应用


Posted in Python onMarch 02, 2016

单例模式的实现方式
将类实例绑定到类变量上

class Singleton(object):
  _instance = None

  def __new__(cls, *args):
    if not isinstance(cls._instance, cls):
      cls._instance = super(Singleton, cls).__new__(cls, *args)
    return cls._instance

但是子类在继承后可以重写__new__以失去单例特性

class D(Singleton):

  def __new__(cls, *args):
    return super(D, cls).__new__(cls, *args)

使用装饰器实现

def singleton(_cls):
  inst = {}

  def getinstance(*args, **kwargs):
    if _cls not in inst:
      inst[_cls] = _cls(*args, **kwargs)
    return inst[_cls]
  return getinstance

@singleton
class MyClass(object):
  pass

问题是这样装饰以后返回的不是类而是函数,当然你可以singleton里定义一个类来解决问题,但这样就显得很麻烦了

使用__metaclass__,这个方式最推荐

class Singleton(type):
  _inst = {}
  
  def __call__(cls, *args, **kwargs):
    if cls not in cls._inst:
      cls._inst[cls] = super(Singleton, cls).__call__(*args)
    return cls._inst[cls]


class MyClass(object):
  __metaclass__ = Singleton

Tornado中的单例模式运用
来看看tornado.IOLoop中的单例模式:

class IOLoop(object):

  @staticmethod
  def instance():
    """Returns a global `IOLoop` instance.

Most applications have a single, global `IOLoop` running on the
main thread. Use this method to get this instance from
another thread. To get the current thread's `IOLoop`, use `current()`.
"""
    if not hasattr(IOLoop, "_instance"):
      with IOLoop._instance_lock:
        if not hasattr(IOLoop, "_instance"):
          # New instance after double check
          IOLoop._instance = IOLoop()
    return IOLoop._instance

为什么这里要double check?来看个这里面简单的单例模式,先来看看代码:

class Singleton(object):

  @staticmathod
  def instance():
    if not hasattr(Singleton, '_instance'):
      Singleton._instance = Singleton()
    return Singleton._instance

在 Python 里,可以在真正的构造函数__new__里做文章:

class Singleton(object):

  def __new__(cls, *args, **kwargs):
    if not hasattr(cls, '_instance'):
      cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
    return cls._instance

这种情况看似还不错,但是不能保证在多线程的环境下仍然好用,看图:

Python设计模式中单例模式的实现及在Tornado中的应用

出现了多线程之后,这明显就是行不通的。

1.上锁使线程同步
上锁后的代码:

import threading

class Singleton(object):

  _instance_lock = threading.Lock()
  
  @staticmethod
  def instance():
    with Singleton._instance_lock:
      if not hasattr(Singleton, '_instance'):
        Singleton._instance = Singleton()
    return Singleton._instance

这里确实是解决了多线程的情况,但是我们只有实例化的时候需要上锁,其它时候Singleton._instance已经存在了,不需要锁了,但是这时候其它要获得Singleton实例的线程还是必须等待,锁的存在明显降低了效率,有性能损耗。

2.全局变量
在 Java/C++ 这些语言里还可以利用全局变量的方式解决上面那种加锁(同步)带来的问题:

class Singleton {

  private static Singleton instance = new Singleton();
  
  private Singleton() {}
  
  public static Singleton getInstance() {
    return instance;
  }
  
}

在 Python 里就是这样了:

class Singleton(object):

  @staticmethod
  def instance():
    return _g_singleton

_g_singleton = Singleton()

# def get_instance():
# return _g_singleton

但是如果这个类所占的资源较多的话,还没有用这个实例就已经存在了,是非常不划算的,Python 代码也略显丑陋……

所以出现了像tornado.IOLoop.instance()那样的double check的单例模式了。在多线程的情况下,既没有同步(加锁)带来的性能下降,也没有全局变量直接实例化带来的资源浪费。

3.装饰器

如果使用装饰器,那么将会是这样:

import functools

def singleton(cls):
  ''' Use class as singleton. '''

  cls.__new_original__ = cls.__new__

  @functools.wraps(cls.__new__)
  def singleton_new(cls, *args, **kw):
    it = cls.__dict__.get('__it__')
    if it is not None:
      return it

    cls.__it__ = it = cls.__new_original__(cls, *args, **kw)
    it.__init_original__(*args, **kw)
    return it

  cls.__new__ = singleton_new
  cls.__init_original__ = cls.__init__
  cls.__init__ = object.__init__

  return cls

#
# Sample use:
#

@singleton
class Foo:
  def __new__(cls):
    cls.x = 10
    return object.__new__(cls)

  def __init__(self):
    assert self.x == 10
    self.x = 15

assert Foo().x == 15
Foo().x = 20
assert Foo().x == 20
def singleton(cls):
  instance = cls()
  instance.__call__ = lambda: instance
  return instance

#
# Sample use
#

@singleton
class Highlander:
  x = 100
  # Of course you can have any attributes or methods you like.

Highlander() is Highlander() is Highlander #=> True
id(Highlander()) == id(Highlander) #=> True
Highlander().x == Highlander.x == 100 #=> True
Highlander.x = 50
Highlander().x == Highlander.x == 50 #=> True
Python 相关文章推荐
python 多进程通信模块的简单实现
Feb 20 Python
Python中用format函数格式化字符串的用法
Apr 08 Python
用Python遍历C盘dll文件的方法
May 06 Python
Python函数可变参数定义及其参数传递方式实例详解
May 25 Python
Python字符串格式化输出方法分析
Apr 13 Python
Python随机生成均匀分布在三角形内或者任意多边形内的点
Dec 14 Python
Python比较2个时间大小的实现方法
Apr 10 Python
Python基于分析Ajax请求实现抓取今日头条街拍图集功能示例
Jul 19 Python
Python 实现子类获取父类的类成员方法
Jan 11 Python
基于python实现自动化办公学习笔记(CSV、word、Excel、PPT)
Aug 06 Python
Python爬虫实战案例之爬取喜马拉雅音频数据详解
Dec 07 Python
django中websocket的具体使用
Jan 22 Python
Python使用设计模式中的责任链模式与迭代器模式的示例
Mar 02 #Python
详解Python设计模式编程中观察者模式与策略模式的运用
Mar 02 #Python
Python设计模式编程中解释器模式的简单程序示例分享
Mar 02 #Python
分析Python中设计模式之Decorator装饰器模式的要点
Mar 02 #Python
实例解析Python设计模式编程之桥接模式的运用
Mar 02 #Python
Python随机生成带特殊字符的密码
Mar 02 #Python
Python设计模式编程中Adapter适配器模式的使用实例
Mar 02 #Python
You might like
php写的带缓存数据功能的mysqli类
2012/09/06 PHP
PHP中使用memcache存储session的三种配置方法
2014/04/05 PHP
php中__toString()方法用法示例
2016/12/07 PHP
javascript 二维数组的实现与应用
2010/03/16 Javascript
基于jQuery的弹出警告对话框美化插件(警告,确认和提示)
2010/06/10 Javascript
javascript获取url上某个参数的方法
2013/11/08 Javascript
js 操作符汇总
2014/11/08 Javascript
node.js中的favicon.ico请求问题处理
2014/12/15 Javascript
js读取并解析JSON类型数据的方法
2015/11/14 Javascript
javascript中apply、call和bind的使用区别
2016/04/05 Javascript
微信小程序 Record API详解及实例代码
2016/09/30 Javascript
Vue.js组件tree实现无限级树形菜单
2016/12/02 Javascript
bootstrap警告框使用方法解析
2017/01/13 Javascript
javascript回调函数的概念理解与用法分析
2017/05/27 Javascript
JavaScript的级联函数用法简单示例【链式调用】
2019/03/26 Javascript
详解JavaScript中精度失准问题及解决方法
2020/02/04 Javascript
整理 node-sass 安装失败的原因及解决办法(小结)
2020/02/19 Javascript
基于JavaScript实现表格隔行换色
2020/05/08 Javascript
python实现爬虫下载美女图片
2015/07/14 Python
python 3.0 模拟用户登录功能并实现三次错误锁定
2017/11/01 Python
人机交互程序 python实现人机对话
2017/11/14 Python
利用python和ffmpeg 批量将其他图片转换为.yuv格式的方法
2019/01/08 Python
Django通过dwebsocket实现websocket的例子
2019/11/15 Python
python3中for循环踩过的坑记录
2020/12/14 Python
酒店工作职员求职简历的自我评价
2013/10/23 职场文书
护士检查书
2014/01/17 职场文书
会计自我鉴定
2014/02/04 职场文书
大学生求职信范文
2014/05/24 职场文书
领导干部整治奢华浪费之风思想汇报
2014/10/07 职场文书
医药公司采购员岗位职责
2015/04/03 职场文书
超市采购员岗位职责
2015/04/07 职场文书
2015年计划生育责任书
2015/05/08 职场文书
婚礼家长致辞
2015/07/27 职场文书
远程教育集中轮训基层干部培训班学习心得体会
2016/01/09 职场文书
python爬虫--selenium模块
2021/03/31 Python
MySQL8.0的WITH查询详情
2021/08/30 MySQL