python单例模式

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()

  可以看到,结果是:

<span>python单例模式</span>

这个过程是,首先__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://javaforall.net/119486.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • fprintf函数和fscanf函数「建议收藏」

    fprintf函数和fscanf函数「建议收藏」1、fscanf函数只能从文本文件中按格式输入。fscanf函数和scanf函数相似,只是输入的对象是磁盘上文本文件中的数据。函数的调用形式如下:fscanf(文件指针,格式控制字符串,输入项表)例如,若文件指针fp已指向一个已打开的文本文件,a、b分别为整型变量,则以下语句从fp所指的文件中读入两个整数放入变量a和b中:fscanf(fp, “%d%d”,&a,&b);注意:文件中的两个整数之间

    2022年10月18日
    4
  • win10运行cmd拒绝访问_如何拒绝不想回答的问题

    win10运行cmd拒绝访问_如何拒绝不想回答的问题尝试以下方法:1)开启|关闭防火墙(这里需要关闭)sudoufwenable|disable2)开启远程服务在终端界面输入:servicesshdstart。 结果显示:ssh:unrecognizedservice。 输入查看命令:servicesshstatus 显示也是unrecognizedservice。 开启远程服务:终端界面键入:sudo/etc/init….

    2025年12月13日
    4
  • 注册境外邮箱有哪些_腾讯会员邮箱登录

    注册境外邮箱有哪些_腾讯会员邮箱登录TOMVIP邮箱可在客户端绑定,支持多个终端邮件同步。163VIP邮箱超大容量、邮件群发效果好、微信邮件实时提醒,企业版邮箱也可以使用误删恢复功能。

    2022年9月23日
    4
  • lookdiv激活码【2021最新】[通俗易懂]

    (lookdiv激活码)JetBrains旗下有多款编译器工具(如:IntelliJ、WebStorm、PyCharm等)在各编程领域几乎都占据了垄断地位。建立在开源IntelliJ平台之上,过去15年以来,JetBrains一直在不断发展和完善这个平台。这个平台可以针对您的开发工作流进行微调并且能够提供…

    2022年3月26日
    229
  • 阿里云申请免费ssl证书及安装_申请免费ssl证书

    阿里云申请免费ssl证书及安装_申请免费ssl证书本文参考以下文章并整理:阿里云SSL证书免费申请方法(图文教程)藏羚骸的博客~阿里云SSL证书部署(DigiCert免费版SSL)2022阿里云免费SSL证书品牌为DigiCertDV单域名证书,每个阿里云账号可以申请20个免费SSL证书资源包,SSL证书大全图文详解阿里云SSL证书免费申请和部署教程,包括SSL证书申请域名DNS验证等操作。阿里云DigiCert免费版SSL有效期一年,过期后需要重新部署SSL。所以,不管是第一次部署SSL还是刚接手公司项目SSL就到期的小伙伴都可

    2022年10月3日
    4
  • matlab中三角函数的反函数怎么表示(反三角函数的对应数值)

    原文地址:Matlab反三角函数表示作者:蓝薇-Matlab反三角函数命令:asin(),acos(),atan(),acot()生成的值为弧度值,即rad;asind(),acosd(),atand(),acotd()生成的值为角度值;将弧转换为角度的命令:rad2deg()deg2rad()作用则相反…

    2022年4月10日
    370

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号