Python 实现单例模式的几种方式以及存在的问题

一、模块单例

Python 的模块就是天然的单例模式,因为模块在第一次导入时,会生成 .pyc 文件,当第二次导入时,就会直接加载 .pyc 文件,而不会再次执行模块代码。

#foo1.py
class Singleton(object):
    def foo(self):
        pass
singleton = Singleton()


#foo.py
from foo1 import singleton

直接在其他文件中导入此文件中的对象,这个对象即是单例模式的对象

二、静态变量方法

先执行了类的__new__方法(我们没写时,默认调用object.new),实例化对象;然后再执行类的__init__方法,对这个对象进行初始化,所有我们可以基于这个,实现单例模式。

class Singleton(object):
    def __new__(cls,a):
        if not hasattr(cls, '_instance'):
            cls._instance = object.__new__(cls)
        return cls._instance
    def __init__(self,a):
        self.a = a
    def aa(self):
        print(self.a)

a = Singleton("a")

变种:利用类的静态方法或者类方法,实现对函数初始化的控制。该方法需要手动调用静态方法实现实例。本质上是手动版的__new__方法。

class Singleton:
    @staticmethod
    def instance1(*args, **kwargs):
        if not hasattr(Singleton, "_instance"):
            Singleton._instance = Singleton(*args, **kwargs)
        return Singleton._instance

  
    @classmethod
    def instance2(cls,*args, **kwargs):
        if not hasattr(cls, "_instance"):
            Singleton._instance = Singleton(*args, **kwargs)
        return Singleton._instance


a = Singleton.instance1() 
b = Singleton.instance2() 
print(a == b)```

## 三、元类方法
此方法是在__new__方法的更上层对实例化过程进行控制。

原理:执行元类的 元类的__new__方法和__init__方法用来实例化类对象,__call__ 方法用来对实例化的对象的实例即类的对象进行控制。__call__方法会调用实例类的 __new__方法,用于创建对象。返回对象给__call__方法,然后调用类对象的 __init__方法,用于对对象初始化。

```python
class Singleton1(type):
    def __init__(self, *args, **kwargs):
        self.__instance = None
        super(Singleton1,self).__init__(*args, **kwargs)

    def __call__(self, *args, **kwargs):
        if self.__instance is None:
            self.__instance = super(Singleton1,self).__call__(*args, **kwargs)
        return self.__instance

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

class C(metaclass=Singleton1):
    pass

四、装饰器

原理:装饰器用来控制类调用__call__方法。

def singleton(cls, *args, **kw):
    instance = {}
    def _singleton(args):
        if cls not in instance:
            instance[cls] = cls(*args, **kw)
        return instance[cls]
    return _singleton

@singleton
class A:
    pass

五、注意

除了模块单例外,其他几种模式的本质都是通过设置中间变量,来判断类是否已经被实例。区别就是中间变量的位置不同,或设置在元类中,或封装在函数中,或设置在类中作为静态变量。

注意1:中间变量的访问和更改存在线程安全的问题:在开启多线程模式的时候需要加锁处理。

注意2:new__方法无法避免触发__init(),初始的成员变量会进行覆盖。其他方法不会。

单例模式是一种常用的设计模式,它保证某个类只有一个实存在,并提供一个全局访问点。在 Python 中,单例模式实现方式有以下几种: 1. 使用模块:Python 的模块是天然的单例模式,因为模块在第一次导入时会被缓存,之后的导入会直接使用缓存中的模块对象。因此,我们可以将需要实现单例模式的类定义在一个模块中,然后在其他模块中导入这个模块即可。 2. 使用 __new__ 方法:Python 中的每个类都有一个特殊的 __new__ 方法,它会在实化对象时被调用。我们可以重写 __new__ 方法,在第一次实化对象时创建一个实,并将其缓存起来,之后的实化直接返回缓存的实即可。 3. 使用装饰器:我们可以编写一个装饰器,将需要实现单例模式的类装饰起来,使其只能实化一次。具体实现可以利用闭包来保存对象,每次实化时判断是否已经实化过,如果是则直接返回对象。 4. 使用元类:元类是用于创建类的类,我们可以编写一个元类,重写其 __call__ 方法,在创建实时判断是否已经存在,如果存在则直接返回对象。 对于这些实现方式,还有一些优化技巧可以提升其效率和可用性,如: 1. 线程安全:如果多个线程同时调用对象,可能会导致创建多个实的情况,因此我们需要加锁来保证线程安全。 2. 序列化和反序列化:如果对象需要进行序列化和反序列化操作,我们需要实现 __getstate__ 和 __setstate__ 方法,以便正确地保存和加载对象。 3. 继承和子类化:如果我们需要从类派生子类,可能会破坏单例模式,因此我们需要重写 __new__ 方法,以便在创建子类对象时也返回对象。 综上所述,单例模式是一种常用的设计模式,在 Python 中有多种实现方式和优化技巧可供选择。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值