网站首页 > 技术文章 正文
目录
- 为什么需要懒加载
- 如何使用懒加载
- 最后的话
懒加载是一种编程范式,它推迟加载操作,直到不得不这样做。通常,当操作开销很大,需要耗费大量时间或空间时,惰性求值是首选实现。
例如,在 Python 中,涉及惰性求值的最著名技术之一是生成器。生成器不是为迭代创建整个序列,而是懒惰地一次生成一个元素。
在 Python 世界之外,许多其他面向对象的编程语言,例如 Swift 和 Kotlin,都具有与对象相关的惰性求值。
具体来说,你可以指定自定义实例对象的特定属性是惰性的,这意味着在显式访问这些属性之前不会创建这些属性。
为什么需要懒加载
在我们开始讨论懒属性之前,有些人可能想知道为什么它很重要,或者我们为什么要使用懒属性。
比如在社交网站中,一个功能是查看一个人的关注者,以列表的形式呈现。当我们点击一个用户时,我们可以在弹出窗口中查看该用户的个人资料。
获取用户个人资料数据的操作可能很昂贵,不仅需要访问远程服务器,还需要将数据存储在内存中。
那么在编程实现时可以把关注者的个人资料作为懒属性,仅在点击特定用户名时才获取该属性。
这就是为什么我们需要懒属性。
如何使用懒加载
方法 1:
使用 @property
@property 是一个装饰器,可以将常规函数转化为属性,比如支持点符号访问。因此,严格来说,创建属性并不是真正创建lan属性本身。相反,它只是提供一个接口来简化数据处理的问题。让我们先看看下面的代码。
class User:
def __init__(self):
self._profile_data = None
@property
def profile_data(self):
if self._profile_data is None:
print("执行耗时操作...")
self._profile_data = 'profile data'
return self._profile_data
demo = User()
print("init done")
print(demo.profile_data)
#init done
#执行耗时操作...
#profile data
初始化完成后并不会执行耗时操作,对应的加载用户列表就不会觉得卡。只有在获取用户资料(点击操作)时,程序会先判断是否已经存在 _profile_data,没有才会执行耗时操作,如果有直接返回,大大提升了效率。
方法 2:
使用 __getattr__ 特殊方法
在 Python 中,名称前后有双下划线的函数称为魔术方法。__getattr__ 可以帮助我们实现懒属性。
对于自定义类,实例对象的属性保存在字典中,可以访问实例对象的 __dict__ 属性获取。值得注意的是,如果__dict__ 不包含指定的属性,Python 将会调用魔术的方法 __getattr__,写个代码你就明白了:
class User:
def __init__(self):
self._profile_data = None
self.name = 'None'
def __getattr__(self, item):
print("called __getattr__")
if item == 'profile_data':
if self._profile_data is None:
print("执行耗时操作...")
self._profile_data = 'profile data'
return self._profile_data
user = User()
print("init done")
print(user.__dict__)
print(user.profile_data)
print(user.__dict__)
print(user.name)
输出结果如下:
init done
{'_profile_data': None, 'name': 'None'}
called __getattr__
执行耗时操作...
profile data
{'_profile_data': 'profile data', 'name': 'None'}
None
和方法 1 一样,初始化完成后并不会执行耗时操作,我们在获取 profile_data 属性时,由于 profile_data 不在 __dict__ 中,因此会执行 __getattr__ 方法获取,而 name 在 __dict__ 获取 name 属性时根本就不会执行 __getattr__ 方法。
怎么判断一个属性是不是在 __dict__ 中呢,只要没有显式的定义该属性,或者使用 setattr 来设置属性,它就不会在 __dict__ 中。
因此可以借助魔术方法 __getattr__ 来创建懒属性 profile_data。
需要注意,Python 还有一个类似的魔术方法 __getattribute__,与 __getattr__ 方法不同的是, 每次获取属性时都会调用 __getattribute__ 方法。
class User:
def __init__(self):
self._profile_data = None
self.name = 'None'
def __getattribute__(self, item):
print("called __getattr__")
user = User()
print("init done")
print(user.profile_data)
print(user.name)
程序输出如下:
init done
called __getattr__
None
called __getattr__
None
此功能仅在你期望属性非常频繁地更改并且只有最新数据相关时才有用。在这些情况下,我们可以通过定义相关函数来实现效果。
最后的话
在本文中,我们重点讨论了在 Python 中实现懒属性的两种实用方法:一种是使用 @property 装饰器,另一种是使用 __getattr__ 特殊方法。
就我个人而言,我更喜欢使用属性装饰器,它更直接、更容易理解。但是,当你需要定义多个懒属性时,该 getattr 方法更好,因为它提供了一个集中的地方来管理这些懒属性。
原文链接:
https://www.jb51.net/article/222893.htm
猜你喜欢
- 2025-06-10 【Python】性能加速之解析器加速Pypy 库使用说明
- 2025-06-10 【Python】性能加速之解析器加速Brython 库使用说明
- 2025-06-10 Rust与Python的文件系统性能对比分析:你可能想知道的一切
- 2025-06-10 比C语言还快20%!Mojo首个大模型开放下载,性能达Python版250倍
- 2025-06-10 Java、Go 和 Python 多线程性能对比
- 2025-06-10 Python 3.13.0 重磅发布:性能起飞,GIL 不再是瓶颈!
- 2025-06-10 精通Python多进程(Multiprocessing)提升性能:8 个进阶层次解析
- 2025-06-10 Python 3.13 启动自由线程,性能会下降吗?
- 2025-06-10 Python3.11性能测评超3.10近64%
- 2025-06-10 Python的lru_cache:提升函数性能的神奇魔法
- 265℃Python短文,Python中的嵌套条件语句(六)
- 264℃python笔记:for循环嵌套。end=""的作用,图形打印
- 263℃PythonNet:实现Python与.Net代码相互调用!
- 259℃Python实现字符串小写转大写并写入文件
- 257℃Python操作Sqlserver数据库(多库同时异步执行:增删改查)
- 117℃原来2025是完美的平方年,一起探索六种平方的算吧
- 98℃Python 和 JavaScript 终于联姻了!PythonMonkey 要火?
- 90℃Ollama v0.4.5-v0.4.7 更新集合:Ollama Python 库改进、新模型支持
- 最近发表
-
- 想要提升Python代码效率?这五个高级方法是必须要会的
- 函数还能返回函数?Python 这个神操作让代码效率翻倍
- 新手学Python避坑,学习效率狂飙! 二十一、print()函数
- 新手学Python避坑,学习效率狂飙!十九、break 与 continue
- 新手学Python避坑,学习效率狂飙!十三、if语句和while语句
- Python自动化办公:打工人常用的10个效率工具,告别加班!
- Python的10个神级冷门技巧,让代码效率翻倍!
- 提升Python编程效率的10点建议(python小技巧及速度提高)
- 90%的Python程序员都不知道的(效率密码 )函数让代码直接封神!
- 【Python】性能加速之解析器加速Pypy 库使用说明
- 标签列表
-
- python中类 (31)
- python 迭代 (34)
- python 小写 (35)
- python怎么输出 (33)
- python 日志 (35)
- python语音 (31)
- python 工程师 (34)
- python3 安装 (31)
- python音乐 (31)
- 安卓 python (32)
- python 小游戏 (32)
- python 安卓 (31)
- python聚类 (34)
- python向量 (31)
- python大全 (31)
- python次方 (33)
- python桌面 (32)
- python总结 (34)
- python浏览器 (32)
- python 请求 (32)
- python 前端 (32)
- python验证码 (33)
- python 题目 (32)
- python 文件写 (33)
- python中的用法 (32)