先看下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__()方法定義為靜態(tài)方法,并且至少需要傳遞一個參數(shù)cls,cls表示需要實例化的類,此參數(shù)在實例化時由Python解釋器自動提供。
我們來看下下面類中對__new__()方法的實現(xiàn):
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...
發(fā)現(xiàn)實例化對象的時候,調(diào)用__init__()初始化之前,先調(diào)用了__new__()方法
__new__()必須要有返回值,返回實例化出來的實例,需要注意的是,可以return父類__new__()出來的實例,也可以直接將object的__new__()出來的實例返回。
__init__()有一個參數(shù)self,該self參數(shù)就是__new__()返回的實例,__init__()在__new__()的基礎(chǔ)上可以完成一些其它初始化的動作,__init__()不需要返回值。
若__new__()沒有正確返回當(dāng)前類cls的實例,那__init__()將不會被調(diào)用,即使是父類的實例也不行。
我們可以將類比作制造商,__new__()方法就是前期的原材料購買環(huán)節(jié),__init__()方法就是在有原材料的基礎(chǔ)上,加工,初始化商品環(huán)節(jié)。
實際應(yīng)用過程中,我們可以這么使用:
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__()方法的使用,就是再進(jìn)行初始化之前,檢查緩存中是否存在該對象,如果存在則將緩存存放對象直接返回,如果不存在,則將對象放至緩存中,供下次使用。
更多信息請查看IT技術(shù)專欄