常见的在Python中实现单例模式的三种方法

        单例模式是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例类的特殊类。通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。

        单例模式的要点有三个;一是某个类只能有一个实二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例在Python中,单例模式有以下几种实现方式。

方法一、实现__new__方法,然后将类的一个实例绑定到类变量_instance上;如果cls._instance为None,则说明该类还没有被实例化过,new一个该类的实例,并返回;如果cls._instance不为None,直接返回_instance,代码如下:

示例一:

class Singleton(object):
  
  def __new__(cls, *args, **kwargs):
    if not hasattr(cls, '_instance'):
      orig = super(Singleton, cls)
      cls._instance = orig.__new__(cls, *args, **kwargs)
    return cls._instance
  
class MyClass(Singleton):
  a = 1
  
one = MyClass()
two = MyClass()
  
#one和two完全相同,可以用id(), ==, is检测
print id(one)  # 29097904
print id(two)  # 29097904
print one == two  # True
print one is two  # True

示例二:

class Dog(object):
    __instance = None
    __init_flag = False

    def __new__(cls,name):    #__new__方法用于创建对象,这里利用类属性进行判断,是否为第一次创建对象
        if cls.__instance == None:      #如果是第一次,则用cls.__instance类属性指向创建的对象的引用
            cls.__instance = object.__new__(cls)
            return cls.__instance           #并返回创建对象的引用
        else:
            return cls.__instance #再一次创建时,cls.__instance已经不是None了,这时else返回第一次创建对象的引用

    def __init__(self,name):      #__init__方法当创建完对象时既会被调用,这里也用类属性__init_flag  = False进行判断
        if self.__init_flag == False:      #第一次初始化时,将输入的参数存入实例对象属性中,并修改类属性__init_flag =True
            self.name = name 
            self.__init_flag = True

a = Dog('wangcai')
print(id(a))
print(a.name)
b = Dog('xiaowangcai')
print(id(b))
print(b.name)

方法二、本质上是方法一的升级版,使用__metaclass__(元类)的高级python用法,具体代码如下:

class Singleton2(type):
  
  def __init__(cls, name, bases, dict):
    super(Singleton2, cls).__init__(name, bases, dict)
    cls._instance = None
  
  def __call__(cls, *args, **kwargs):
    if cls._instance is None:
      cls._instance = super(Singleton2, cls).__call__(*args, **kwargs)
    return cls._instance
  
class MyClass2(object):
  __metaclass__ = Singleton2
  a = 1
  
one = MyClass2()
two = MyClass2()
  
print id(one)  # 31495472
print id(two)  # 31495472
print one == two  # True
print one is two  # True

方法三、使用Python的装饰器(decorator)实现单例模式,这是一种更Pythonic的方法;单利类本身的代码不是单例的,通装饰器使其单例化,代码如下:

def singleton(cls, *args, **kwargs):
  instances = {}
  def _singleton():
    if cls not in instances:
      instances[cls] = cls(*args, **kwargs)
    return instances[cls]
  return _singleton
  
@singleton
class MyClass3(object):
  a = 1
  
one = MyClass3()
two = MyClass3()
  
print id(one)  # 29660784
print id(two)  # 29660784
print one == two  # True
print one is two  # True

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值