python的单例模式--解决多线程的单例模式失效
单例模式(Singleton Pattern)
是⼀种常⽤的软件设计模式,主要⽬的是确保某⼀个类只有⼀个实例存在。希望在整个系统中,某个类只能出现⼀个实例时,单例对象就能派上⽤场
⽐如,某个服务器程序的配置信息存放在⼀个⽂件中,客户端通过⼀个 AppConfig 的类来读取配置⽂件的信息。如果在程序运⾏期间,有很多地⽅都需要使⽤配置⽂件的内容,也就是说,很多地⽅都需要创建 AppC 使⽤需求: 程序运⾏起来只需要⼀份
⽐如: admin的register
数据库连接,数据库连接池
(全局变量)
全局变量 --- 模块导⼊ (使⽤同⼀个数据)
单例模式
单例模式的应⽤---数据库连接池
class SingleDBpool(object):
def __init__(self):
self.pool = ...
def __new__(cls, *args, **kwargs):
if not hasattr(cls,'_instance'):
cls._instance = super(SingleDBpool,cls).__new__(*args, **kwargs)
return cls._instance
def connect(self):
return tion()
Python
1 使⽤模块
Python 的模块就是天然的单例模式,因为模块在第⼀次导⼊时,会⽣成 .pyc ⽂件,当第⼆次导⼊时,就会直接加载 .pyc ⽂件,⽽不会再次执⾏模块代码。因此,我们只需把相关的函数和
数据定义在⼀个模块中,就可以获得⼀个单例对象了。
# mysingleton.py
class My_Singleton(object):
def foo(self):
pass
my_singleton = My_Singleton()
将上⾯的代码保存在⽂件 mysingleton.py 中,然后这样使⽤:
from mysingleton import my_singleton
my_singleton.foo()
2 使⽤ new
为了使类只能出现⼀个实例,我们可以使⽤ new 来控制实例的创建过程,代码如下:
class Singleton(object):
_instance = None
def __new__(cls, *args, **kw):
if not cls._instance:
cls._instance = super(Singleton, cls).__new__(cls, *args, **kw)
return cls._instance
class MyClass(Singleton):
a = 1
3 使⽤类⽅法但是每次调⽤会很繁琐 A.get_instance(params) -- ⽆法⽀持多线程
class A(object):
instance = None
def __init__(self,name):
self.name = name
@classmethod
def get_instance(cls,*args,**kwargs):
if not cls.instance:
cls.instance = cls(*args,**kwargs)
return cls.instance
a = A.get_instance('aaa')
b = A.get_instance('bbb')
print(a.name)
多线程下,为什么会失效?
import threading
class Single(object):
instance = None
def __init__(self):
import time
time.sleep(0.5) # 有延时的情况
pass
@classmethod
def get_instance(cls,*args,**kwargs):
if not cls.instance:
cls.instance = cls(*args,**kwargs)
return cls.instance
def task(arg):
obj = _instance()
print(obj)
for i in range(5):
t = threading.Thread(target=task,args=[i,])
t.start()
结果:创建了不同的对象 -- 失效
# <__main__.Single object at 0x00000000029B8F60>
# <__main__.Single object at 0x00000000029B8E80>
# <__main__.Single object at 0x000000000299BD68>
# <__main__.Single object at 0x00000000029F2BE0>
# <__main__.Single object at 0x00000000029C2B38>
如何解决多线程下单例的失效
>> 加线程锁
import threading
import time
class Single(object):
instance = None
_threading_lock = threading.Lock()
def __init__(self):
time.sleep(0.5)
@classmethod
def get_instance(cls,*args,**kwargs):
if not cls.instance: # 先判断是否存在(如果存在,说明不是多线程,直接获取) with cls._threading_lock: # 加锁,只有⼀个线程进⼊,然后判断单例是否存在 if not cls.instance: # 先判断是否存在(如果存在,说明不是多线程,直接获取) cls.instance = cls(*args,**kwargs)
return cls.instance
return cls.instance
def task(arg):
obj = _instance()
print(obj)
for i in range(5):
t = threading.Thread(target=task,args=[i,])
t.start()
time.sleep(5)
obj = _instance()
print(obj)
4 使⽤装饰器
可以使⽤装饰器来装饰某个类,使其只能⽣成⼀个实例
def Singleton(cls):
instance = []
def inner(*args,**kwargs):
if cls(*args,**kwargs) not in instance:
instance.append(cls(*args,**kwargs))
return instance[0]
return inner
@Singleton
class A(object):
pass
a = A()
b = A()
print(a == b)
5 使⽤ metaclass
对象和类创建的完整流程:
class F:
pass
1 执⾏type的 init ⽅法(类是type的对象)
obj = F()
2 执⾏type的 call ⽅法
2.1 调⽤ F类的 new ⽅法 (创建对象)
2.2 调⽤ F类的 init ⽅法 (对象初始化)
obj()
3 执⾏ F的 call ⽅法
# 继承 type 类(模拟重写 type -- ⽤于创建类)
class Single(type):
def __init__(self,*args,**kwargs):
super(Single,self).__init__(*args,**kwargs)
def __call__(cls, *args, **kwargs):
obj = cls.__new__(cls,*args, **kwargs)
cls.__init__(obj,*args, **kwargs)
return obj
# ⽤伪type 创建Foo类
class Foo(metaclass=Single): # 通过 Single 创建
def __init__(self,name):
self.name= name
def __new__(cls, *args, **kwargs):
return object.__new__(cls,*args, **kwargs)
元类(metaclass)可以控制类的创建过程,它主要做三件事:
拦截类的创建
修改类的定义
返回修改后的类
使⽤元类实现单例模式:
class Single(type): # 通过
def __call__(cls, *args, **kwargs):java单例模式双重锁
if not hasattr(cls,'_instance'):
cls._instance = cls.__new__(cls,*args, **kwargs)
return cls._instance
class Foo(metaclass=Single): # 通过 Single 创建
pass
这样通过创建Single类,以后需要单例模式的类,可以指定⽤它来创建就可以了
Python 的模块是天然的单例模式,这在⼤部分情况下应该是够⽤的,也可以使⽤装饰器、元类等⽅法metaclass补充
class MyType(type):
def __init__(self, *args, **kwargs):
super(MyType, self).__init__(*args, **kwargs)
def __call__(cls, *args, **kwargs):
return super(MyType, cls).__call__(*args, **kwargs)
def with_metaclass(base):
return MyType('XX', (base,), {})
class Foo(with_metaclass(object)):
pass
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论