前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布

单例

作者头像
用户1733462
发布2018-06-01 17:30:05
4920
发布2018-06-01 17:30:05
举报
文章被收录于专栏:数据处理数据处理
new实现单例

new至少要有一个参数cls,代表要实例化的类,new方法负责创建一个实例对象,在对象被创建的时候调用该方法它是一个类方法,new方法负责创建一个实例对象,在对象被创建的时候调用该方法它是一个类方法。new方法在返回一个实例之后,会自动的调用init方法,对实例进行初始化。如果new方法不返回值,或者返回的不是实例,那么它就不会自动的去调用init方法。此参数在实例化时由Python解释器自动提供

代码语言:javascript
复制
class Singleton(object):
    def __new__(cls, *args, **kwargs):
        print cls,'xxxx'
        if not hasattr(cls,"_instance"):
            cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
        
        return cls._instance
元类实现单例
代码语言:javascript
复制
class Singleton(type):
    def __init__(self, *args, **kwargs):
        print "__init__"
        self.__instance = None
        super(Singleton,self).__init__(*args, **kwargs)

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


class Foo(object):
    __metaclass__ = Singleton #在代码执行到这里的时候,元类中的__new__方法和__init__方法其实已经被执行了,而不是在Foo实例化的时候执行。且仅会执行一次。只有再需要对原类做变动时才需要写new方法

为了更好理解上面单例,注意下面两种写法

代码语言:javascript
复制
class UpperAttrMetaClass(type):
    
    def __new__(upperattr_metaclass, future_class_name, future_class_parents, future_class_attr):
        print 'xxxxxxx'
        uppercase_attr = {}
        for name, value in future_class_attr.items():
            if not name.startswith('__'):
                uppercase_attr[name.upper()] = value
            else:
                uppercase_attr[name] = value
        return type.__new__(upperattr_metaclass,future_class_name, future_class_parents, uppercase_attr)#返回一个对象,但同时这个对象是一个类
    
    def __init__(upperattr_metaclass,*args, **kwargs):
        print 'init'
        upperattr_metaclass.__instance = None
        super(UpperAttrMetaClass,upperattr_metaclass).__init__(*args, **kwargs)
    def myprint(self, param):
        print param,'xxxxx'

    def __call__(upperattr_metaclass,*args, **kwargs):
        print 'call'
    #  return upperattr_metaclass
        if upperattr_metaclass.__instance is None:
            upperattr_metaclass.__instance = super(UpperAttrMetaClass,upperattr_metaclass ).__call__(*args, **kwargs)
        return upperattr_metaclass.__instance

下面两种写法 ,

代码语言:javascript
复制
class Test(object):
    __metaclass__ = UpperAttrMetaClass
    def __call__(self):
        print 'test call'
    def __new__(cls, *args, **kwargs):
        print 'test new'
        return super(Test, cls).__new__(cls,*args, **kwargs)
    def __init__(self):
        print 'test init'
    def myprint(self, param):
        print param
    aaaa = 22222
代码语言:javascript
复制
class Test(object):
    def __call__(self):
        print 'test call'
    def myprint(self, param):
        print param
    aaaa = 22222

Test = UpperAttrMetaClass('Test', (object,), {'myprint':Test().myprint, '__call__':Test().__call__, 'aaaa':Test().aaaa}) 

这里我们重写了call方法,一开始不理解

代码语言:javascript
复制
super(UpperAttrMetaClass,upperattr_metaclass ).__call__(*args, **kwargs)

为什么能得到实例,如果我们不重写call方法,Test()将得到实例对象,那么重写的话,重写的格式就是这样,在Test类中重写new,init方法这时被调用。

代码语言:javascript
复制
t = Test()

t.MYPRINT('xxx')
t2 = Test()
print t 
print t2.AAAA
print Test()
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2017.10.24 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • new实现单例
  • 元类实现单例
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档