標籤:python __new__
先看下object類中對__new__()方法的定義:
class object: @staticmethod # known case of __new__ def __new__(cls, *more): # known special case of object.__new__ """ T.__new__(S, ...) -> a new object with type S, a subtype of T """ pass
object將__new__()方法定義為靜態方法,並且至少需要傳遞一個參數cls,cls表示需要執行個體化的類,此參數在執行個體化時由Python解譯器自動提供。
我們來看下下面類中對__new__()方法的實現:
class Demo(object): def __init__(self): print '__init__() called...' def __new__(cls, *args, **kwargs): print '__new__() - {cls}'.format(cls=cls) return object.__new__(cls, *args, **kwargs)if __name__ == '__main__': de = Demo()
輸出:
__new__() - <class '__main__.Demo'>__init__() called...
發現執行個體化對象的時候,調用__init__()初始化之前,先調用了__new__()方法
__new__()必須要有返回值,返回執行個體化出來的執行個體,需要注意的是,可以return父類__new__()出來的執行個體,也可以直接將object的__new__()出來的執行個體返回。
__init__()有一個參數self,該self參數就是__new__()返回的執行個體,__init__()在__new__()的基礎上可以完成一些其它初始化的動作,__init__()不需要返回值。
若__new__()沒有正確返回當前類cls的執行個體,那__init__()將不會被調用,即使是父類的執行個體也不行。
我們可以將類比作製造商,__new__()方法就是前期的原材料購買環節,__init__()方法就是在有原材料的基礎上,加工,初始化商品環節。
實際應用過程中,我們可以這麼使用:
class LxmlDocument(object_ref): cache = weakref.WeakKeyDictionary() __slots__ = ['__weakref__'] def __new__(cls, response, parser=etree.HTMLParser): cache = cls.cache.setdefault(response, {}) if parser not in cache: obj = object_ref.__new__(cls) cache[parser] = _factory(response, parser) return cache[parser]
該類中的__new__()方法的使用,就是再進行初始化之前,檢查緩衝中是否存在該對象,如果存在則將緩衝存放對象直接返回,如果不存在,則將對象放至緩衝中,供下次使用。
python中的__new__()方法