python之懒惰属性(延迟初始化)

  Python 对象的延迟初始化是指,当它第一次被建立时才进行初始化,或者保存第一次建立的结果,而后每次调用的时候直接返回该结果。延迟初始化主要用于提升性能,避免浪费计算,并减小程序的内存需求。python

1. 温故下property

  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 propertyspa

2.lazy property实现

  实现延迟初始化有两种方式,一种是使用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)
View Code
相关文章
相关标签/搜索