函数装饰器和类装饰器实现单例类

单例类,指的是这个类只能创建一个实例,创建完成后,其他类实例都无法再创建。今天我们来看一下,使用函数装饰器和类装饰器怎么实现这种特殊一点的类。

函数装饰器实现

装饰器算是类里面比较难的内容之一,但是实际上它的思想并不复杂。简单点说,就是在你原来内容的基础上,在外面给你加点东西,实现类似装饰的效果。但是它是怎么实现的呢?一般来说,都是通过拦截函数调用来实现的,比如:用装饰器装饰函数的时候,它拦截函数调用,装饰类的时候,它拦截类实例的创建调用,即拦截类初始化__init__函数。
知道这个原理以后,我们就可以来尝试实现了。

首先来看通过函数装饰器拦截类的创建过程,代码如下:

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

def singleIns(cls):
    def onCall(*args):
        return createInstance(cls, *args)
    return onCall

上面就是这个函数装饰器singleIns的实现,它返回一个函数调用,当用它来装饰一个类,创建类实例的时候,就会用onCall方法拦截类的__init__方法。我们再来看一下它怎么使用的。

@singleIns
class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age
    def __str__(self):
        return "{} 的年龄是{}".format(self.name, self.age)
zhangsan = Person(‘zhangsan‘, 30)
lisi = Person(‘lisi‘, 29)
print(zhangsan)
print(lisi)

最终的输出结果是:

zhangsan 的年龄是30
zhangsan 的年龄是30

为什么结果是一样的?因为在创建实例的过程中,__init__函数被onCall函数拦截,此时会进入到createInstance函数的流程中,会对这个类实例进行判断,如果不存在这个类的实例,那么就初始化一个后返回,如果存在,直接返回第一个创建的类实例。因此最终只有一个类实例存在,就实现类单例类。

类装饰器实现

上面最开始的位置我们说了,函数装饰器和类装饰器都是拦截函数调用,在函数装饰器实现类调用拦截的地方我们看到,它是通过函数装饰器内部的函数来实现拦截的。如果是类装饰器呢,它通过什么来拦截呢?
答案是call函数来拦截,我们来看一下类装饰器的实现代码:

class singleIns:
    def __init__(self, cls):
        self.cls = cls
        self.ins = None
    def __call__(self, *args):
        if self.ins is None:
            self.ins = self.cls(*args)
        return self.ins

代码和函数装饰器相比,其实功能没有太多变化,通过__call__方法来接收被拦截类的初始化函数参数args,然后用args来初始化类实例。但是只在这个类还没有实例的情况下进行初始化,否则直接返会初始化好的类。
我们来看一下应用的代码:

@singleIns
class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age
    def __str__(self):
        return "{} 的年龄是{}".format(self.name, self.age)
zhangsan = Person(‘zhangsan‘, 30)
lisi = Person(‘lisi‘, 29)
print(zhangsan)
print(lisi)

最终的结果和上面函数装饰器的一样,如下所示:

zhangsan 的年龄是30
zhangsan 的年龄是30

以上就是通过函数装饰器和类装饰器实现单例类的两个简单实例,仅做抛砖引玉之说,如果有其他想法或者疑问,可以给我留言一起探讨。

原文地址:https://blog.51cto.com/14730652/2475866

时间: 2025-01-31 06:53:00

函数装饰器和类装饰器实现单例类的相关文章

C++单例模式与单例类模板

1.单例模式 (1)需求的提出:在架构设计时,某些类在整个系统生命周期中最多只能有一个对象存在(Single,Instance) (2)问题:如何定义一个类,使得这个类最多只能创建一个对象? 要控制类的对象数目,必须对外隐藏构造函数 思路: @1:将构造函数的访问属性设置为private,将拷贝构造函数,赋值操作符都声明成private的,防止对象的赋值操作,因为这个类的对象只能有一个. @2:定义instance并初始化为NULL.在类中用static type* instance的方式声明一

单例类多线程

作为设计模式理论中的Helloworld,相信学习java语言的人,都应该听说过单例模式.单例模式作为对象的一种创建模式,它的作用是确保某一个类在整个系统中只有一个实例,而且自行实例化并向整个系统提供这个实例. 由此可见,单例模式具有以下的特点: 单例类只能有一个实例. 单例类必须自己创建自己的唯一的实例. 单例类必须给所有其他对象提供这一实例. 由于Java语言的特点,使得单例模式在Java语言的实现上有自己的特点.这些特点主要表现在单例类如何将自己实例化. 饿汉式单例类 饿汉式单例类是在Ja

Java中Class和单例类的作用与类成员的理解

Java中Class类的作用与深入理解 在程序运行期间,Java运行时系统始终为所有的对象维护一个被称为运行时的类型标识.这个信息跟踪着每个对象所属的类.JVM利用运行时信息选择相应的方法执行.而保存这些信息的类称为Class.可能容易产生混淆,容易想到class.不过二者没什么关系,class不过是描述类的一个关键字.而Class却是保存着运行时信息的类. 它能做什么?Class类可以帮助我们在程序运行时分析类,说白了就是获取类中的值.可能瞬间就想到了反射,没错!Class一般就是和反射配套使

设计模式中饿汉式单例类和懒汉式单例类

单例模式的特点为: *单例类只有一个实例. *单例类必须自己创建自己的唯一实例. *单例类必须给所有其他对象提供这一实例. 饿汉式单例类: 而饿汉式单例类则在java语言中是最为简单的单例类,这是一个描述饿汉式单例类的类图的实现. 此类图中,此类已经将自己实例化. 源码为: package singleton; public class EagerSingleton { private static EagerSingleton instance=new EagerSingleton(); /*

单例类的总结

/** * 单例类的两种方法. * 1.饿汉式 * 当类一加载就开始创建对象 * 2.懒汉式 * 当需要使用到类的对象是调用类的类方法才开始创建对象,想比较上面而言是 * 懒汉式. * 注意,单例类的特点: * 1.必须要将构造函数私有化,这样才能防止别的类通过构造方法来建立对象,但是也是因为如此,所以 * 单例类的成员变量和成员函数都必须要用static修饰,因此要想调用单例类的函数,只能通过类名.类方法的方式来进行 * 调用. * 2.必须要在类内创建一个本类的对象. * 3.向外提供一个公

第62课 单例类模板

1. 单例需求的提出 (1)在架构设计时,某些类在整个系统生命期中最多只能有一个对象存在(Single Instance) (2)问题:如何定义一个类,使得这个类最多只能创建一个对象 2. 单例模式 (1)要控制类的对象数目,必须对外隐藏构造函数 (2)思路 ①将构造函数的访问属性设置为private ②定义instance,并初始化为NULL ③提供静态函数getInstance来返回单例实例,并判断当instance的值为空时,并创建对象,并保存在instance指针中,非空值时直接返回in

(七)boost库之单例类

(七)boost库之单例类 一.boost.serialzation的单件实现 单例模式是一种常用的软件设计模式.在它的核心结构中只包含一个被称为单例类的特殊类.通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源.如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案. 单例,通常在一个大型项目中单例是非常常见的,boost库没有提供专门的单例类,但可以在其它库中找到他的实现 #include <boost/serializat

单例类模板(五十一)

在有些时候,我们在进行架构设计时,某些类在整个系统生命期中最多只能有一个对象存在(Single Instance).比如在超市的收银系统,一个扫描机就对应一个,每个屏幕也会对应一个,基于这种情况,我们就提出了单例模式.那么问题就来了:如何定义一个类,并且使得这个类最多只能创建一个对象呢?那么就必须得控制类的对象数目,必须对外隐藏构造函数.思路是:将构造函数的访问属性设置为 private,定义 instance 并初始化为 NULL:当需要使用对象时,访问 instance 的值,如果是空值,那

28.C++- 单例类模板(详解)

描述 指在整个系统生命期中,一个类最多只能有一个实例(instance)存在,使得该实例的唯一性(实例是指一个对象指针)  , 比如:统计在线人数 在单例类里,又分为了懒汉式和饿汉式,它们的区别在于创建实例的时间不同: 懒汉式 : 指代码运行后,实例并不存在,只有当需要时,才去创建实例(适用于单线程) 饿汉式 : 指代码一运行,实例已经存在,当时需要时,直接去调用即可(适用于多线程) 用法 将构造函数的访问属性设置为private, 提供一个GetInstance()静态成员函数,只能供用户访问