python单例模式

一、单例模式

单例模式是应用开发过程中最简单同时也是最著名的一种创建型设计模式。单例模式提供了一个类只有一个特定类型的对象的机制。

通常应用于场景为:日志记录、数据库操作等,程序运行时只能生成一个实例,避免对同一资源产生冲突的访问请求。

二、如何设计单例模式

(1)重写构造函数__new__方法

class Singleton(object):
    def __new__(self):
        if not hasattr(self,'instance'):
            self.instance = super(Singleton,self).__new__(self)
        return self.instance
        

a = Singleton()
b = Singleton()


print id(a),id(a)

  该方法主要思路就是在覆盖实例化函数__new__(),在其中添加判断,检查对象是否存在。hasattr是python的特殊方法,用来查看对象是否具有某个属性。

当然,如果想要继承这个Singleton,子类中一定不要忘了重写__new__方法,否则会覆盖掉父类中已经修改的方法。

class Singleton(object):
    def __new__(self):
        if not hasattr(self,'instance'):
            self.instance = super(Singleton,self).__new__(self)
        return self.instance
        
class test(Singleton):
    def __new__(self):
        super(Singleton,self).__new__(self)
a = test()
b = test()


print id(a),id(a)

  

(2)元类编程重写__call__方法

元类是一个类的类,可以通过创建元类,重新定义类的行为。当我们用类A创建一个类的时候,python通过A = type( name , bases , dict )创建它,其中name是类的名称,base是基类,dict是属性变量。

下面讨论元类type构造类以及实例的过程:

class Meta(type):
    def __init__(self, name, bases, attrs):
        print "init"
        super(Meta, self).__init__(name, bases, attrs)
    def __new__(meta,name, bases, attrs):
        print "new"
        return super(Meta,meta).__new__(meta,name, bases, attrs)
    def __call__(self):
        print "call"
        return super(Meta,self).__call__()
        
        
s = ["Student", (object, ), {"name": "Joe", "age": 25}   ]    
ClassStudent = Meta(*s)
instance = ClassStudent()

  可以看到,结果是:

这个过程是,首先__new__创建类的实例,作为__init__的输入,完成构造函数的作用。当然这个过程的结果是获得了一个实例并且是叫做:

Student的类,ClassStudent是对这个类的引用。当采用()调用,即实现Meta的实例的可调用的实例化过程。

至此,我们简单了解了相关的魔术函数的作用。

 

通过元类实现单例模式的方法为:

class Meta(type):
    _instance = {}
    def __init__(self, name, bases, attrs):
        print "init"
        super(Meta, self).__init__(name, bases, attrs)
    def __new__(meta,name, bases, attrs):
        print "new"
        return super(Meta,meta).__new__(meta,name, bases, attrs)
    def __call__(self):
        print "call"
        
        if self not in self._instance:
            self._instance[self] = super(Meta,self).__call__()
        return self._instance[self]
        
        
s = ["Student", (object, ), {"name": "Joe", "age": 25}   ]    
ClassStudent = Meta(*s)
instance1 = ClassStudent()
instance2 = ClassStudent()
print id(instance1),id(instance2)

  至此,可以看到,通过元类编程实现单例模式,需要改的是__call__函数,因为其构造函数是为了构造类的,而不是在生成实例的过程中,通过函数调用的方式生成实例会调用__call__,因此在这个位置操作。

 当然,对于单例类的使用,可以在类中通过__metaclass__属性来设置,取代用type类直接生成的方法。

class Meta(type):
    _instance = {}
    def __init__(self, name, bases, attrs):
        print "init"
        super(Meta, self).__init__(name, bases, attrs)
    def __new__(meta,name, bases, attrs):
        print "new"
        return super(Meta,meta).__new__(meta,name, bases, attrs)
    def __call__(self):
        print "call"
        
        if self not in self._instance:
            self._instance[self] = super(Meta,self).__call__()
        return self._instance[self]
 

class ClassStudent(object):
    __metaclass__ = Meta
    pass
       
instance1 = ClassStudent()
instance2 = ClassStudent()
print id(instance1),id(instance2)

  当用户定义一个类class的时候,Python解释器就会在当前类中查找"__metaclass__"属性,如果找到,就通过该属性对应的代码创建类;如果没有找到,就继续以相同的规则查找父类。如果在任何父类中都找不到"__metaclass__",就会用内置的type来创建类对象。

当然对于python3来说,添加了新的声明单例的方法:

class Test(metaclass = MyMeta):
    pass

  

三、Monostate模式

对于上述单例模式,关注的是生成单一的实例,而通常程序员需要的是让实例共享相同的状态,因此,有时候需要关注状态和行为,而不是同一性。这种概念即为Monostate(单态)模式。python实现这个模式的过程较为轻松:

class ClassStudent(object):
    _age = 20
    pass
       
instance1 = ClassStudent()
instance2 = ClassStudent()
print id(instance1),id(instance2)
print instance1._age,instance2._age

  可以看到,虽然不是一个实例,但是状态相同。

 

转载于:https://www.cnblogs.com/pinking/p/7820498.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Python 中,单例模式可以通过使用模块、装饰器或元类来实现。下面分别介绍这三种实现方式。 1. 使用模块 Python 中的模块是单例的,因为模块在程序中只会被导入一次,之后再次导入时,会直接使用已经存在的模块对象。可以利用这个特性来实现单例模式,将需要实现单例的类定义在一个模块中,然后导入该模块即可。 例如,假设我们有一个名为 `MySingleton` 的类,我们可以将其定义在一个名为 `mysingleton.py` 的模块中,并在需要使用单例的地方导入该模块。这样,无论在哪里导入该模块,都是使用同一个 `MySingleton` 实例,从而实现单例模式。 2. 使用装饰器 可以使用装饰器来将一个类变成单例模式。具体实现方式是:定义一个装饰器函数,在该装饰器函数中创建一个字典,用于保存已经创建的实例。在调用被装饰的类时,先检查该字典中是否已经存在该类的实例,如果存在,则返回已经存在的实例,否则创建新的实例,并保存到字典中。 下面是一个使用装饰器实现单例模式的示例: ```python def singleton(cls): instances = {} def get_instance(*args, **kwargs): if cls not in instances: instances[cls] = cls(*args, **kwargs) return instances[cls] return get_instance @singleton class MySingleton: def __init__(self, x): self.x = x obj1 = MySingleton(1) obj2 = MySingleton(2) print(obj1.x) # 输出 1 print(obj2.x) # 输出 1 print(obj1 is obj2) # 输出 True ``` 在这个例子中,我们定义了一个名为 `singleton` 的装饰器函数,该函数接受一个类作为参数,并返回一个函数 `get_instance`。在 `get_instance` 函数中,我们首先检查字典 `instances` 中是否已经存在该类的实例。如果已经存在,则直接返回该实例,否则创建新的实例,并保存到字典中。 在定义 `MySingleton` 类时,我们使用了 `@singleton` 装饰器来将它变成单例模式。在调用 `MySingleton` 类时,实际上是调用了 `get_instance` 函数,从而实现了单例模式。 3. 使用元类 可以使用元类来定义一个类的创建方式,从而实现单例模式。具体实现方式是:定义一个继承自 `type` 的元类,重写 `__call__` 方法,在该方法中检查是否已经存在该类的实例,如果存在,则返回已经存在的实例,否则创建新的实例,并保存到字典中。 下面是一个使用元类实现单例模式的示例: ```python class Singleton(type): instances = {} def __call__(cls, *args, **kwargs): if cls not in cls.instances: cls.instances[cls] = super().__call__(*args, **kwargs) return cls.instances[cls] class MySingleton(metaclass=Singleton): def __init__(self, x): self.x = x obj1 = MySingleton(1) obj2 = MySingleton(2) print(obj1.x) # 输出 1 print(obj2.x) # 输出 1 print(obj1 is obj2) # 输出 True ``` 在这个例子中,我们定义了一个名为 `Singleton` 的元类,该元类继承自 `type` 类,并重写了 `__call__` 方法。在 `__call__` 方法中,我们首先检查字典 `instances` 中是否已经存在该类的实例。如果已经存在,则直接返回该实例,否则创建新的实例,并保存到字典中。 在定义 `MySingleton` 类时,我们使用了 `metaclass` 参数来指定该类的元类为 `Singleton`,从而将它变成单例模式。在调用 `MySingleton` 类时,实际上是调用了 `Singleton` 元类的 `__call__` 方法,从而实现了单例模式。 ### 回答2: Python单例模式是一种设计模式,用于确保一个类只有一个实例存在。在Python中,单例模式通常通过使用装饰器或元类来实现。 装饰器的实现方式是在类定义之前定义一个装饰器函数,该函数用于创建并返回一个实例。当类的实例被创建时,装饰器函数会检查是否已经存在一个实例,如果存在则返回该实例,否则创建一个新实例并返回。 元类的实现方式是定义一个继承自`type`的元类,并重写`__call__`方法。该方法在实例创建时被调用,可以在此方法中实现单例逻辑。在该方法中,可以保存一个类的实例,并在后续调用时返回该实例,确保只有一个实例存在。 无论是使用装饰器还是元类,单例模式都遵循以下原则: 1. 只能有一个实例存在,多次创建实例只返回同一个实例。 2. 实例可以全局访问,无论在哪个模块中,都可以通过调用类的方法获取实例。 3. 确保线程安全,避免多个线程同时创建实例。 由于Python的动态特性,单例模式Python中相对容易实现。但需要注意的是,单例模式可能导致全局状态的共享,增加了组件之间的耦合性,所以在使用时需谨慎考虑。在不明确需要使用单例模式的情况下,建议优先考虑其他的设计模式。 ### 回答3: 单例模式是一种创建对象的设计模式,它确保类只有一个实例,并提供全局访问点。 在Python中,实现单例模式有多种方法,下面分别介绍两种常见的方式。 1. 使用模块实现单例模式: 在Python中,模块在首次被导入时,会被解析并执行模块中的代码。因此,我们可以将需要实现单例模式的类定义在一个模块中,并在其他模块中导入该模块,确保该类只被实例化一次。示例代码如下: ```python # singleton.py class SingletonClass: def __init__(self, name): self.name = name singleton_instance = SingletonClass("Singleton Instance") # main.py from singleton import singleton_instance def main(): print(singleton_instance.name) if __name__ == "__main__": main() ``` 以上代码中,`SingletonClass`是一个需要实现单例模式的类,我们在`singleton.py`中将其实例化为`singleton_instance`,然后在`main.py`中导入并使用这个实例。 2. 使用装饰器实现单例模式: 另一种常见的实现方式是使用装饰器,通过装饰器将一个类包装成单例模式。示例代码如下: ```python def singleton(cls): instances = {} def wrapper(*args, **kwargs): if cls not in instances: instances[cls] = cls(*args, **kwargs) return instances[cls] return wrapper @singleton class SingletonClass: def __init__(self, name): self.name = name def main(): instance1 = SingletonClass("Singleton Instance 1") instance2 = SingletonClass("Singleton Instance 2") print(instance1.name) print(instance2.name) if __name__ == "__main__": main() ``` 以上代码中,我们定义了一个名为`singleton`的装饰器函数,将需要实现单例模式的类`SingletonClass`作为参数传递给装饰器。在`wrapper`函数中,通过判断类是否已经存在实例,如果不存在则创建一个新的实例,并将其存储到`instances`字典中。每次调用`SingletonClass`时,实际上是调用了`wrapper`函数并返回相应的实例。 通过以上两种方式,我们可以实现Python中的单例模式,确保某个类只有一个实例。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值