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
<p>@lru_cache(maxsize=128)
def _get_user_prefs_cached(user_id: int) -> dict:</p><h1>模拟耗时操作,如查 DB 或远程调用</h1><pre class="brush:php;toolbar:false;">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
<p>_user_cache_var = contextvars.ContextVar("user_cache", default={})</p><div class="aritcle_card flexRow">
                                                        <div class="artcardd flexRow">
                                                                <a class="aritcle_card_img" href="/ai/1464" title="Cliclic AI"><img
                                                                                src="https://img.php.cn/upload/ai_manual/000/000/000/175680372963805.png" alt="Cliclic AI"  onerror="this.onerror='';this.src='/static/lhimages/moren/morentu.png'" ></a>
                                                                <div class="aritcle_card_info flexColumn">
                                                                        <a href="/ai/1464" title="Cliclic AI">Cliclic AI</a>
                                                                        <p>Cliclic商品背景图编辑器是一款功能强大的AI工具,帮助用户快速生成具有吸引力的商品图背景。</p>
                                                                </div>
                                                                <a href="/ai/1464" title="Cliclic AI" class="aritcle_card_btn flexRow flexcenter"><b></b><span>下载</span> </a>
                                                        </div>
                                                </div><p>def get_cached_user_data(key: str, factory_func, *args, *<em>kwargs) -> Any:
cache = _user_cache_var.get()
if key not in cache:
cache[key] = factory_func(</em>args, **kwargs)
_user_cache_var.set(cache)
return cache[key]</p><h1>在依赖中使用:</h1><p>def get_user_report(current_user = Depends(get_current_user)):
return get_cached_user<em>data(
f"report</em>{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 用户的缓存结果。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

183

2024.05.11

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

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

226

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,显著提升吞吐量和响应速度,尤其适用于处理数据库查询、网络请求等耗时操作,无需阻塞主线程。

28

2025.12.22

Python 微服务架构与 FastAPI 框架
Python 微服务架构与 FastAPI 框架

本专题系统讲解 Python 微服务架构设计与 FastAPI 框架应用,涵盖 FastAPI 的快速开发、路由与依赖注入、数据模型验证、API 文档自动生成、OAuth2 与 JWT 身份验证、异步支持、部署与扩展等。通过实际案例,帮助学习者掌握 使用 FastAPI 构建高效、可扩展的微服务应用,提高服务响应速度与系统可维护性。

251

2026.02.06

if什么意思
if什么意思

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

847

2023.08.22

登录token无效
登录token无效

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

6630

2023.09.14

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

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

843

2023.09.14

token怎么获取
token怎么获取

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

1092

2023.12.21

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

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

精品课程

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

共6课时 | 0.4万人学习

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

共72课时 | 7.2万人学习

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

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