Python中的__new__()方法的使用


__new__() 函數只能用於從object繼承的新式類。

 

先看下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__()方法的使用,就是再進行初始化之前,檢查緩存中是否存在該對象,如果存在則將緩存存放對象直接返回,如果不存在,則將對象放至緩存中,供下次使用。

 

再來個單例的,通過重載__new__實現單例:

class Singleton(object):
    def __new__(cls, *args, **kwargs):
        if not hasattr(cls, '_instance'):
            cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
        return cls._instanc


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM