python中@lazy_property
来源:互联网 发布:visio中文版mac下载 编辑:程序博客网 时间:2024/06/07 09:42
原文: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描述符的话,可以参考Descriptor HowTo Guide。
方式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
方法,说明确实延迟初始化了。
- python中@lazy_property
- python中运行其他python
- 【Python语法】python中__name__
- Python中numpy中asarray
- Python中numpy中asarray
- python中list中方法
- Python学习中...
- Python中使用中文
- C中调用PYTHON
- 学习Python中
- python中使用getopt
- python学习中
- python中编码转换
- python中%符号详解
- Python技巧 ---- 不断更新中
- python中使用getopt
- python中合并文件
- python中enumerate用法
- ubuntu sudo apt-get update 失败 解决方法
- UBUNTU如何安装tar.gz版的flash
- ASP.NET MVC 5 - 添加搜索
- 第十周—C语言 输出10000以内的回文数
- libevent 学习笔记 一
- python中@lazy_property
- CABasicAnimation fillMode和removedOnCompletion
- dll 调用,动态,静态
- 我的技术博客地址
- leetcode_58. Length of Last Word 字符串最后一个单词的长度,字符串分词
- 剑指Offer(面试题11-13)
- angularJS中input相关指令详解
- 图像增强之01概述
- 移动端mixin的px转rem