博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
python中@lazy_property
阅读量:2433 次
发布时间:2019-05-10

本文共 2183 字,大约阅读时间需要 7 分钟。

原文:https://segmentfault.com/a/1190000005818249

Python 对象的延迟初始化是指,当它第一次被创建时才进行初始化,或者保存第一次创建的结果,然后每次调用的时候直接返回该结果。

延迟初始化主要用于提高性能,避免浪费计算,并减少程序的内存需求。

property

在切入正题之前,我们了解下property的用法,property可以将属性的访问转变成方法的调用。

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

lazy property

实现延迟初始化有两种方式,一种是使用python描述符,另一种是使用@property修饰符。

方式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__方法。

__get__()方法中,调用实例的area()方法计算出结果,并动态给实例添加一个同名属性area,然后将计算出的值赋予给它,相当于设置c.__dict__['area']=val

当我们再次调用c.area时,直接从c.__dict__中进行查找,这时就会直接返回之前计算好的值了。

不太懂python描述符的话,可以参考。

方式2

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_propertyclass 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类似。

我们可以检查下是否真的延迟初始化了:

c = Circle(4) print "before first visit"print c.__dict__  c.areaprint "after first visit"print c.__dict__

输出结果为:

before first visit{
'radius': 4}evaluteafter first visit{
'_lazy_area': 50.24, 'radius': 4}

从中可以看出,只有当我们第一次访问c.area时,才调用area方法,说明确实延迟初始化了。

转载地址:http://zjlmb.baihongyu.com/

你可能感兴趣的文章
HashMap的remove()方法详解
查看>>
单例模式-分解步骤,逐步解析
查看>>
通过Form表单一次性拿到json格式数据,及后台接收
查看>>
## EL表达式与JSTL标签用法解读
查看>>
Mybatis异常:The content of elements must consist of well-formed.......(一般出现在写分页/带大于小于号的SQL)
查看>>
Mybatis光速入门(配置文件模块)
查看>>
关于Oracle的主键自增如何设置
查看>>
手撕HashMap的resize()方法源码渗透解析+图解
查看>>
Mybatis常见异常类型Could not set parameters for mapping离不开这个原因!
查看>>
Thymeleaf中一个页面怎么嵌套另一个页面,关于页面嵌套,标签告诉你应该知道的
查看>>
JAVA如何实现短信验证码--阿里云接口,新手式图文教学,个人项目有这一篇就够了
查看>>
Java中大小数BigDecimal的加减乘除用法及场景的详细介绍,看完不信你还会报Syntax error on token “+/-/*“, invalid AssignmentOperat异常
查看>>
UVa 10917 Dijkstra
查看>>
CF403B/CF402D
查看>>
CF402E / 403C
查看>>
cf404b
查看>>
cf404c
查看>>
cf404d
查看>>
武大网络预赛 Problem 1545 - I - Twenty-four
查看>>
ZOJ Problem Set - 3768 Continuous Login
查看>>