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