Python 對象的延遲初始化是指,當它第一次被建立時才進行初始化,或者保存第一次建立的結果,而後每次調用的時候直接返回該結果。延遲初始化主要用於提升性能,避免浪費計算,並減小程序的內存需求。python
property能夠將屬性的訪問轉變成方法的調用ide
class Circle(object): def __init__(self, radius): self.radius = radius @property def area(self): return 3.14 * self.radius ** 2 c = Circle(4) print c.radius print c.area
能夠看到,area
雖然是定義成一個方法的形式,可是加上@property
後,能夠直接執行c.area
,當成屬性訪問。性能
如今問題來了,每次調用c.area
,都會計算一次,太浪費cpu了,怎樣才能只計算一次呢?這就是lazy property
spa
實現延遲初始化有兩種方式,一種是使用python描述符,另外一種是使用@property
修飾符code
方法1:對象
class lazy(object): def __init__(self, func): self.func = func def __get__(self, instance, cls): val = self.func(instance) setattr(instance, self.func.__name__, val) return val class Circle(object): def __init__(self, radius): self.radius = radius @ lazy def area(self): print 'evalute' return 3.14 * self.radius ** 2 c = Circle(4) print c.radius print c.area print c.area print c.area
結果'evalute'
只輸出了一次。在lazy
類中,咱們定義了__get__()
方法,因此它是一個描述符。當咱們第一次執行c.area
時,python解釋器會先從c.__dict__
中進行查找,沒有找到,就從Circle.__dict__
中進行查找,這時由於area
被定義爲描述符,因此調用__get__
方法。blog
在__get__()
方法中,調用實例的area()
方法計算出結果,並動態給實例添加一個同名屬性area
,而後將計算出的值賦予給它,至關於設置c.__dict__['area']=val
。內存
當咱們再次調用c.area
時,直接從c.__dict__
中進行查找,這時就會直接返回以前計算好的值了。get
方法2:it
def lazy_property(func): attr_name = "_lazy_" + func.__name__ @property def _lazy_property(self): if not hasattr(self, attr_name): setattr(self, attr_name, func(self)) return getattr(self, attr_name) return _lazy_property class Circle(object): def __init__(self, radius): self.radius = radius @lazy_property def area(self): print 'evalute' return 3.14 * self.radius ** 2
這裏與方法1殊途同歸,在area()
前添加@lazy_property
至關於運行如下代碼:
lazy_property(area)
lazy_property()方法返回_lazy_property
,_lazy_property
又會調用_lazy_property()
方法,剩下的操做與方法1相似。
#性能差方法 class Circle(object): def __init__(self, radius): self.radius = radius @property def area(self): print("come in") return 3.14 * self.radius ** 2 c = Circle(4) print(c.radius) print(c.area) print(c.area) #方法1 class LazyProperty: def __init__(self, method): self.method = method def __get__(self, instance, cls): if not instance: return None value = self.method(instance) setattr(instance,self.method.__name__,value) return value class Circle(object): def __init__(self, radius): self.radius = radius @LazyProperty def area(self): print("come in") return 3.14 * self.radius ** 2 c = Circle(4) print(c.radius) print(c.area) print(c.area) #方法2 def LazyProperty(func): attr_name = "_lazy_" + func.__name__ @property def wrap(self): if not hasattr(self, attr_name): setattr(self, attr_name, func(self)) return getattr(self, attr_name) return wrap class Circle(object): def __init__(self, radius): self.radius = radius @LazyProperty def area(self): print("come in") return 3.14 * self.radius ** 2 c = Circle(4) print(c.radius) print(c.area) print(c.area)