当前位置: 首页 > news >正文

Python functools 模块的 @lru_cache 装饰器介绍

functools.lru_cache 是 Python 标准库 functools 模块中的一个装饰器,用于实现简单的缓存机制。它通过缓存函数的返回值来提高函数的执行效率,特别是对于那些被多次调用且参数相同的函数。

LRU 缓存机制

  • LRU 代表 Least Recently Used,即最近最少使用。

  • LRU 缓存机制会保存最近使用的缓存项,并在缓存满时丢弃最久未使用的项。

lru_cache 装饰器的参数

  • maxsize:指定缓存中可以存储的最大键值对数量,类型:intNone

    • 如果 maxsize=None,则缓存大小无限制。

    • 如果指定了 maxsize,则缓存会限制为该大小。当缓存满时,最久未使用的缓存项会被丢弃(LRU 策略)。

  • 默认值 128 表示缓存可以存储最多 128 个键值对。

  • 每个键值对包括:

    • :函数的参数(包括位置参数和关键字参数)。

    • :函数的返回值。

  • typed:

    • 类型:bool

    • 默认值:False

    • 描述:如果设置为 True,则会区分不同类型的参数。例如,f(3)f(3.0) 会被视为不同的调用。

使用示例

示例 1:缓存斐波那契数列
import functools
import time

@functools.lru_cache(maxsize=None)
def fibonacci(n):
    if n < 2:
        return n
    return fibonacci(n - 1) + fibonacci(n - 2)

# 记录开始时间
start_time = time.time()

print(fibonacci(30))  # 计算斐波那契数列的第 30 项

# 记录结束时间
end_time = time.time()

# 计算运行时间
run_time = end_time - start_time
print(f"加了@lru_cache 装饰器的fibonacci 运行时间: {run_time} 秒")


def fibonacci(n):
    if n < 2:
        return n
    return fibonacci(n - 1) + fibonacci(n - 2)

print(fibonacci(30))  # 计算斐波那契数列的第 30 项

# 记录开始时间
start_time = time.time()

print(fibonacci(30))  # 计算斐波那契数列的第 30 项

# 记录结束时间
end_time = time.time()

# 计算运行时间
run_time = end_time - start_time
print(f"没有 @lru_cache 装饰器的fibonacci 运行时间: {run_time} 秒")

解释

  • 没有缓存时,计算斐波那契数列的第 30 项会非常慢,因为有大量的重复计算。

  • 使用 lru_cache 后,函数的返回值被缓存,重复调用时直接从缓存中获取结果,大大提高了效率。

示例 2:缓存单位四元数

Python复制

import functools
import torch

@functools.lru_cache(maxsize=None)
def identity_quats(batch_dims, dtype=None, device=None, requires_grad=True):
    quat = torch.zeros(
        (*batch_dims, 4), 
        dtype=dtype, 
        device=device, 
        requires_grad=requires_grad
    )
    with torch.no_grad():
        quat[..., 0] = 1
    return quat

# 调用函数
quats = identity_quats((2, 3), dtype=torch.float32, device=torch.device('cpu'), requires_grad=True)
print(quats)

输出

plaintext复制

tensor([[[1., 0., 0., 0.],
         [1., 0., 0., 0.],
         [1., 0., 0., 0.]],

        [[1., 0., 0., 0.],
         [1., 0., 0., 0.],
         [1., 0., 0., 0.]]], grad_fn=<CopySlices>)

解释

  • 第一次调用 identity_quats 时,函数会计算并缓存结果。

  • 后续调用时,如果参数相同,直接从缓存中返回结果,避免重复计算。

缓存内容和生命周期

缓存的内容
  • 最终返回值lru_cache 只缓存函数的最终返回值。它不会保存函数执行过程中的中间状态或中间值。

  • 键值对:缓存是以键值对的形式存储的,其中键是函数的参数(包括位置参数和关键字参数),值是函数的返回值。

缓存的生命周期
  • 内存中保存:缓存的值会一直保留在内存中,直到程序退出,或者缓存被手动清除。

  • 手动清除缓存:可以使用 cache_clear 方法手动清除缓存。例如:

    fibonacci.cache_clear()

    这将清除 fibonacci 函数的所有缓存值。

总结

functools.lru_cache 是一个非常有用的装饰器,用于缓存函数的返回值,从而提高函数的执行效率。它特别适用于那些被多次调用且参数相同的函数。通过合理使用 lru_cache,可以显著提高程序的性能。

相关文章:

  • wps字符很分散
  • 【STM32】SPI通信协议W25Q64Flash存储器芯片(学习笔记)
  • OSS Browser2.0安装使用(阿里云对象存储OSS 图形化界面工具2.0版本)
  • 集成学习(下):Stacking集成方法
  • iPaaS集成平台安全通信的挑战与保障策略
  • MyBatis之参数传递
  • 网络故障排查指南:分治法与排除法结合的分层诊断手册
  • Diamond软件的使用--(5)查看原语
  • 【HTTP 传输过程中的 cookie】
  • 基于springboot和vue项目使用docker部署项目到服务器
  • 【DeepSeek 学c++】dynamic_cast 原理
  • Linux驱动学习笔记(六)
  • HAL库中串口中断开启
  • 23种设计模式(扩展)
  • Elasticsearch 倒排索引 和 正排索引
  • 健康养生:拥抱生活,从呵护身心开始
  • [OpenCV】相机标定之棋盘格角点检测与绘制
  • 集成学习(Ensemble Learning)基础知识2
  • 2025-03-21 Unity 网络基础3——TCP网络通信准备知识
  • 对比4款远程控制软件,画质、性能、功能谁全面?
  • 紫光集团原董事长赵伟国一审被判死缓
  • “异常”只停留在医院里,用艺术为“泡泡宝贝”加油
  • 安徽省委常委、合肥市委书记费高云卸任副省长职务
  • 广东早熟荔枝“抢滩”上海,向长三角消费者喊话:包甜,管够
  • 中拉论坛部长级会议为何悬挂海地和圣卢西亚的国旗?外交部回应
  • 为惩戒“工贼”,美国编剧工会“痛下杀手”