0

0

fastapi 如何让依赖函数支持缓存但按用户隔离

舞夢輝影

舞夢輝影

发布时间:2026-01-23 19:54:09

|

420人浏览过

|

来源于php中文网

原创

FastAPI中实现用户级缓存需将user.id等唯一标识纳入缓存键:①用lru_cache装饰内部函数并传user.id;②用contextvars实现单请求隔离;③生产环境推荐Redis+用户命名空间+TTL。

fastapi 如何让依赖函数支持缓存但按用户隔离

缓存依赖函数但按用户隔离的关键点

FastAPI 本身不提供带用户上下文的缓存机制,cache 类装饰器(比如 functools.lru_cache)是全局共享的,无法区分 current_user。必须手动把用户标识(如 user.iduser.token)纳入缓存键,且确保依赖函数能访问到当前用户。

lru_cache 手动构造用户级缓存键

不能直接装饰依赖函数,因为 lru_cache 在模块加载时就绑定,而 current_user 是运行时注入的。正确做法是:在依赖函数内部调用一个带参数的缓存函数,并把 user.id 作为显式参数传入。

示例:

from functools import lru_cache
from fastapi import Depends, HTTPException

@lru_cache(maxsize=128) def _get_user_prefs_cached(user_id: int) -> dict:

模拟耗时操作,如查 DB 或远程调用

return {"theme": "dark", "lang": "zh"}

def get_user_prefs(current_user = Depends(get_current_user)): if not current_user: raise HTTPException(401) return _get_user_prefs_cached(current_user.id)

  • _get_user_prefs_cached 是真正被缓存的函数,参数必须是可哈希的(intstr 等),不能传 user 对象本身
  • 依赖函数 get_user_prefs 不加缓存装饰,只负责提取 user.id 并转发
  • 注意 maxsize 要合理,避免内存堆积;若用户量大,考虑用 redis 替代 lru_cache

contextvars + 自定义缓存容器(适合更复杂场景)

当需要动态控制缓存生命周期(例如请求结束自动清理)、或缓存值依赖多个上下文变量(如 user.id + request.headers["X-Region"])时,lru_cache 就不够用了。可用 contextvars.ContextVar 绑定每请求独立的缓存字典。

示例:

import contextvars
from typing import Dict, Any

_user_cache_var = contextvars.ContextVar("user_cache", default={})

抖云猫AI论文助手
抖云猫AI论文助手

一款AI论文写作工具,最快 2 分钟,生成 3.5 万字论文。论文可插入表格、代码、公式、图表,依托自研学术抖云猫大模型,生成论文具备严谨的学术专业性。

下载

def get_cached_user_data(key: str, factory_func, *args, *kwargs) -> Any: cache = _user_cache_var.get() if key not in cache: cache[key] = factory_func(args, **kwargs) _user_cache_var.set(cache) return cache[key]

在依赖中使用:

def get_user_report(current_user = Depends(get_current_user)): return get_cached_userdata( f"report{current_user.id}", generate_report_for_user, current_user.id )

  • 每个请求有独立的 cache 字典,天然按用户/请求隔离
  • 需配合中间件在请求开始前初始化 _user_cache_var,否则可能复用上一个请求的缓存
  • 不适用于跨请求复用(比如同一用户多次请求想复用),它只在单次请求内有效

Redis 缓存 + 用户 ID 命名空间(生产推荐)

真实项目中,lru_cachecontextvars 都受限于进程内存和生命周期。用 Redis 可实现跨进程、带 TTL、支持剔除策略的用户级缓存。

关键点:

  • 缓存键必须包含用户唯一标识,例如 f"user:{user.id}:prefs"f"dep:user_prefs:{user.id}"
  • 避免硬编码前缀,建议封装成函数:make_cache_key("user_prefs", user.id)
  • 务必设 ex(TTL),防止脏数据长期滞留;对敏感数据,删除时机要明确(如用户登出时清 Redis 键)
  • 如果依赖函数抛异常,别让错误结果也被缓存——加 try/except 控制是否写入缓存

FastAPI 依赖中调用 Redis 示例(使用 aioredis):

async def get_user_settings(
    current_user = Depends(get_current_user),
    redis = Depends(get_redis_client)
):
    key = f"user:{current_user.id}:settings"
    cached = await redis.get(key)
    if cached:
        return json.loads(cached)
    data = await fetch_from_db(current_user.id)  # 实际逻辑
    await redis.set(key, json.dumps(data), ex=300)  # 5 分钟过期
    return data

用户隔离不是加个 @cache 就能解决的事。核心永远是:缓存键里有没有稳定、唯一、可预测的用户标识,以及这个标识能不能在缓存读写时被可靠拿到。漏掉任意一环,就会出现 A 用户看到 B 用户的缓存结果。

相关专题

更多
什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

178

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

214

2025.12.18

Python FastAPI异步API开发_Python怎么用FastAPI构建异步API
Python FastAPI异步API开发_Python怎么用FastAPI构建异步API

Python FastAPI 异步开发利用 async/await 关键字,通过定义异步视图函数、使用异步数据库库 (如 databases)、异步 HTTP 客户端 (如 httpx),并结合后台任务队列(如 Celery)和异步依赖项,实现高效的 I/O 密集型 API,显著提升吞吐量和响应速度,尤其适用于处理数据库查询、网络请求等耗时操作,无需阻塞主线程。

27

2025.12.22

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

764

2023.08.22

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6107

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

813

2023.09.14

token怎么获取
token怎么获取

获取token值的方法:1、小程序调用“wx.login()”获取 临时登录凭证code,并回传到开发者服务器;2、开发者服务器以code换取,用户唯一标识openid和会话密钥“session_key”。想了解更详细的内容,可以阅读本专题下面的文章。

1064

2023.12.21

token什么意思
token什么意思

token是一种用于表示用户权限、记录交易信息、支付虚拟货币的数字货币。可以用来在特定的网络上进行交易,用来购买或出售特定的虚拟货币,也可以用来支付特定的服务费用。想了解更多token什么意思的相关内容可以访问本专题下面的文章。

1284

2024.03.01

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

10

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号