Python 延迟初始化(lazy property)
admin
2023-07-31 00:46:06
0

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

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

property

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

1234567891011 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:

1234567891011121314151617181920212223 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

12345678910111213141516171819 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相当于运行以下代码:

1 lazy_property(area)

lazy_property()方法返回_lazy_property_lazy_property又会调用_lazy_property()方法,剩下的操作与方法1类似。

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

123456 c = Circle(4) print \”before first visit\”print c.__dict__  c.areaprint \”after first visit\”print c.__dict__

输出结果为:

12345 before first visit{\’radius\’: 4}evaluteafter first visit{\’_lazy_area\’: 50.24, \’radius\’: 4}

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

参考文献

  • Descriptor HowTo Guide
  • lazy evaluation
  • python中的property及实现lazy property(原博客已找不到)


相关内容

热门资讯

Mobi、epub格式电子书如... 在wps里全局设置里有一个文件关联,打开,勾选电子书文件选项就可以了。
定时清理删除C:\Progra... C:\Program Files (x86)下面很多scoped_dir开头的文件夹 写个批处理 定...
500 行 Python 代码... 语法分析器描述了一个句子的语法结构,用来帮助其他的应用进行推理。自然语言引入了很多意外的歧义,以我们...
scoped_dir32_70... 一台虚拟机C盘总是莫名奇妙的空间用完,导致很多软件没法再运行。经过仔细检查发现是C:\Program...
65536是2的几次方 计算2... 65536是2的16次方:65536=2⁶ 65536是256的2次方:65536=256 6553...
小程序支付时提示:appid和... [Q]小程序支付时提示:appid和mch_id不匹配 [A]小程序和微信支付没有进行关联,访问“小...
pycparser 是一个用... `pycparser` 是一个用 Python 编写的 C 语言解析器。它可以用来解析 C 代码并构...
微信小程序使用slider实现... 众所周知哈,微信小程序里面的音频播放是没有进度条的,但最近有个项目呢,客户要求音频要有进度条控制,所...
Apache Doris 2.... 亲爱的社区小伙伴们,我们很高兴地向大家宣布,Apache Doris 2.0.0 版本已于...
python清除字符串里非数字... 本文实例讲述了python清除字符串里非数字字符的方法。分享给大家供大家参考。具体如下: impor...