zoukankan      html  css  js  c++  java
  • python三种方法创建单例模式

    #  方法一,装饰器
    from functools import wraps
    
    
    def single_instance(cls):
        """装饰器实现单例模式,装饰器相当于把类给包了起来,强制返回一个对象"""
        _instance = None
    
        @wraps(cls)
        def wrapper(*args, **kwargs):
            nonlocal _instance
            if _instance is None:
                _instance = cls(*args, **kwargs)
            return _instance
    
        return wrapper
    
    
    @single_instance
    class Test1:
        def __init__(self, name):
            self.name = name
            print("Test init..., name=%s" % self.name)
    
    
    # 方法二  __new__方法
    class Test2:
        _instance = None
    
        def __init__(self, name):
            self.name = name
            print("Test2 init..., name=%s" % self.name)
    
        def __new__(cls, *args, **kwargs):
            """__new__主要做的是给实例分配空间,__init__则是在该空间上进行初始化,既然是单例模式,那么空间地址肯定一样,这里做限制"""
            print('__new__, args=', args, ',kwargs=', kwargs)
            if cls._instance is None:
                cls._instance = super().__new__(cls)
            return cls._instance
    
    
    # 方法三 元类
    class SingleType(type):
        """
        普通类相当于是元类的实例,所以在类完成定义时其实是调用元类的__init__方法,而生成类的实例,如 Test('AIF')则是调用元类的__call__方法(debug可以跟踪到元类的__call__最后还是调用普通类的__init__方法)
        这个跟普通类和实例的关系其实是一样的。如 t = Test('AIF'), 生成t会调用Test的__init__方法,而t()则会调用Test的__call__方法
        """
        def __init__(self, *args, **kwargs):
            self._instance = None
            super().__init__(*args, **kwargs)
    
        def __call__(self, *args, **kwargs):
            if self._instance is None:
                self._instance = super().__call__(*args, **kwargs)
            return self._instance
    
    
    class Test3(metaclass=SingleType):
        def __init__(self, name):
            self.name = name
            print("Test3 init..., name=%s" % self.name)
    
    
    if __name__ == '__main__':
        """测试,Test1, Test2, Test3"""
        a = Test3('AIF')
        b = Test3('AIF333')
        print('a=b is %s, id(a)=%s, id(b)=%s' % (a is b, id(a), id(b)))
        print('a.name=%s, b.name=%s' % (a.name, b.name))
  • 相关阅读:
    git this exceeds GitHub's file size limit of 100.00 MB
    使用vue-cli创建vue工程
    【转】Visual Studio Code必备插件
    linux安装openssl
    Centos7离线安装mysql8
    使用nmon来按频率采集数据
    Mac下编译android4.0.4遇到的问题
    32位ubuntu16.4编译android4.1.1
    vmvare安装vmtools菜单灰色
    Substrate 使用
  • 原文地址:https://www.cnblogs.com/yeteng/p/11731059.html
Copyright © 2011-2022 走看看