先看下object類中對__new__()方法的定義:
1
2
3
4
5
|
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):
1
2
3
4
5
6
7
8
|
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() |
輸出:
1
2
|
__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)用過程中,我們可以這么使用:
1
2
3
4
5
6
7
8
9
|
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)行初始化之前,檢查緩存中是否存在該對象,如果存在則將緩存存放對象直接返回,如果不存在,則將對象放至緩存中,供下次使用。