详解python实现线程安全的单例模式


Posted in Python onMarch 05, 2018

单例模式是一种常见的设计模式,该模式的主要目的是确保某一个类只有一个实例存在。当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场。

比如,服务器的配置信息写在一个文件中online.conf中,客户端通过一个 Config 的类来读取配置文件的内容。如果在程序运行期间,有很多地方都需要使用配置文件的内容,那么每个调用配置文件的地方都会创建 Config的实例,这就导致系统中存在多个Config 的实例对象,在配置文件内容很多的情况下,我们就浪费了大量的内存做了同样的事。事实上,对于Config类我们在程序运行期间时只需要一个实例对象即可,这时单例模式就是最好的选择。

python的模块就是天然的单例模式,这里我们使用修饰器来实现单例模式,以下是代码实现

def Singleton(cls):
 instances = {}

 def get_instance(*args, **kw):
  if cls not in instances:
   instances[cls] = cls(*args, **kw)
  return instances[cls]

 return get_instance

代码也很简单,将类传入单例修饰器中,如果该类还未生成实例(instances中不存在该类),那么就生成一个新的实例返回,并记录在instances中。如果已经instances中已经存在该类,那么直接返回实例instances[cls]。

那么这段代码是完美的吗?答案是否定的,这段代码不是线程安全的。要实现线程安全需要配合锁的使用,只有占有锁的线程才能继续访问单例实例,看来我们需要再写一个修饰器来实现线程安全了,以下是完整的代码实现和简单的多线程测试用例。

#!/usr/bin/python
# -*- coding: utf-8 -*-
import threading

def synchronized(func):
 func.__lock__ = threading.Lock()

 def synced_func(*args, **kws):
  with func.__lock__:
   return func(*args, **kws)

 return synced_func

def Singleton(cls):
 instances = {}

 @synchronized
 def get_instance(*args, **kw):
  if cls not in instances:
   instances[cls] = cls(*args, **kw)
  return instances[cls]

 return get_instance

def worker():
 single_test = test()
 print "id----> %s" % id(single_test)

@Singleton
class test():
 a = 1
if __name__ == "__main__":
 task_list = []
 for one in range(30):
  t = threading.Thread(target=worker)
  task_list.append(t)
 for one in task_list:
  one.start()
 for one in task_list:
  one.join()

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
一个基于flask的web应用诞生 记录用户账户登录状态(6)
Apr 11 Python
Python中用字符串调用函数或方法示例代码
Aug 04 Python
Python中turtle作图示例
Nov 15 Python
python安装教程
Feb 28 Python
Python实现将数据框数据写入mongodb及mysql数据库的方法
Apr 02 Python
如何在Cloud Studio上执行Python代码?
Aug 09 Python
使用Python matplotlib作图时,设置横纵坐标轴数值以百分比(%)显示
May 16 Python
python filecmp.dircmp实现递归比对两个目录的方法
May 22 Python
Python3实现英文字母转换哥特式字体实例代码
Sep 01 Python
Python多分支if语句的使用
Sep 03 Python
python简单实现插入排序实例代码
Dec 16 Python
Python卷积神经网络图片分类框架详解分析
Nov 07 Python
分析python动态规划的递归、非递归实现
Mar 04 #Python
python3.x上post发送json数据
Mar 04 #Python
python数据封装json格式数据
Mar 04 #Python
Python爬虫实例扒取2345天气预报
Mar 04 #Python
Python爬虫设置代理IP的方法(爬虫技巧)
Mar 04 #Python
浅析python实现scrapy定时执行爬虫
Mar 04 #Python
Python使用Scrapy爬虫框架全站爬取图片并保存本地的实现代码
Mar 04 #Python
You might like
PHP读取MySQL数据代码
2008/06/05 PHP
Discuz Uchome ajaxpost小技巧
2011/01/04 PHP
php dirname(__FILE__) 获取当前文件的绝对路径
2011/06/28 PHP
js输入框邮箱自动提示功能代码实现
2013/12/10 Javascript
JavaScript开发人员的10个关键习惯小结
2014/12/05 Javascript
jquery实现标签支持图文排列带上下箭头按钮的选项卡
2015/03/14 Javascript
Bootstrap导航栏各元素操作方法(表单、按钮、文本)
2015/12/28 Javascript
Javascript将数字转化成为货币格式字符串
2016/06/22 Javascript
jQuery Easyui DataGrid点击某个单元格即进入编辑状态焦点移开后保存数据
2016/08/15 Javascript
Ionic默认的Tabs模板使用实例
2016/08/29 Javascript
javascript基本数据类型及类型检测常用方法小结
2016/12/14 Javascript
bootstrap 下拉多选框进行多选传值问题代码分析
2017/02/14 Javascript
详解Vue的computed(计算属性)使用实例之TodoList
2017/08/07 Javascript
极简主义法编写JavaScript类
2017/11/02 Javascript
vue绑定的点击事件阻止冒泡的实例
2018/02/08 Javascript
vue2.0 自定义 饼状图 (Echarts)组件的方法
2018/03/02 Javascript
Vue无限滑动周选择日期的组件的示例代码
2018/07/18 Javascript
puppeteer库入门初探
2019/01/09 Javascript
vue+高德地图写地图选址组件的方法
2019/05/18 Javascript
python dict 字典 以及 赋值 引用的一些实例(详解)
2017/01/20 Python
Python爬虫天气预报实例详解(小白入门)
2018/01/24 Python
详解django中url路由配置及渲染方式
2019/02/25 Python
浅谈Python2之汉字编码为unicode的问题(即类似\xc3\xa4)
2019/08/12 Python
Python产生一个数值范围内的不重复的随机数的实现方法
2019/08/21 Python
python利用opencv保存、播放视频
2020/11/02 Python
中国专业的音频分享平台:喜马拉雅
2019/05/24 全球购物
建筑个人求职信范文
2014/01/25 职场文书
老人节标语大全
2014/10/08 职场文书
房地产销售员岗位职责
2015/04/11 职场文书
预备党员转正党小组意见
2015/06/01 职场文书
《珍珠鸟》教学反思
2016/02/16 职场文书
2016年安康杯竞赛活动总结
2016/04/05 职场文书
《曾国藩家书》读后感——读家书,立家风
2019/08/21 职场文书
导游词之镇江-金山寺
2019/10/14 职场文书
python开发制作好看的时钟效果
2022/05/02 Python
go goth封装第三方认证库示例详解
2022/08/14 Golang