title: FastAPI依赖注入与上下文管理

date: 2025/04/07 00:28:04

updated: 2025/04/07 00:28:04

author: cmdragon

excerpt:

FastAPI框架依赖注入与上下文管理实战指南详细介绍了全局依赖配置、应用生命周期管理和综合应用案例。全局依赖用于统一处理认证、日志、数据库会话等跨路由逻辑,支持多层级配置。应用生命周期管理通过lifespan事件实现资源初始化和释放。电商系统案例展示了如何结合数据库和缓存进行商品创建操作。常见报错解决方案提供了针对数据库连接、请求验证等问题的排查与预防措施。

categories:

  • 后端开发
  • FastAPI

tags:

  • FastAPI
  • 依赖注入
  • 上下文管理
  • 全局依赖
  • 生命周期管理
  • 数据库会话
  • 错误处理


扫描二维码关注或者微信搜一搜:编程智域 前端至全栈交流与成长

探索数千个预构建的 AI 应用,开启你的下一个伟大创意

FastAPI框架依赖注入与上下文管理实战指南

1. 全局依赖配置原理与实现

1.1 全局依赖的核心作用

全局依赖是FastAPI实现跨路由通用逻辑的关键机制,其核心作用包括:

  • 统一处理认证鉴权
  • 标准化响应格式
  • 集中收集请求日志
  • 管理数据库会话生命周期
  • 实施统一速率限制
from fastapi import Depends, FastAPI, Header

app = FastAPI()

async def verify_token(authorization: str = Header(...)):
if not authorization.startswith("Bearer "):
raise HTTPException(status_code=401)
return authorization[7:] app = FastAPI(dependencies=[Depends(verify_token)])

1.2 多层级依赖配置

FastAPI支持灵活的依赖注入层级:

层级类型 作用范围 典型应用场景
全局依赖 所有路由 身份认证、请求日志
路由组依赖 指定路由组 API版本控制、权限分级
单路由依赖 单个路由 特殊参数校验、业务级权限

1.3 数据库会话实战案例

from contextlib import asynccontextmanager
from sqlalchemy.ext.asyncio import AsyncSession, create_async_engine
from sqlalchemy.orm import sessionmaker DATABASE_URL = "postgresql+asyncpg://user:password@localhost/db" engine = create_async_engine(DATABASE_URL)
async_session = sessionmaker(engine, class_=AsyncSession, expire_on_commit=False) @asynccontextmanager
async def lifespan(app: FastAPI):
# 应用启动时执行
async with engine.begin() as conn:
await conn.run_sync(Base.metadata.create_all)
yield
# 应用关闭时执行
await engine.dispose() async def get_db():
async with async_session() as session:
try:
yield session
await session.commit()
except Exception:
await session.rollback()
raise app = FastAPI(lifespan=lifespan, dependencies=[Depends(get_db)])

2. 应用生命周期管理

2.1 生命周期事件实战

from fastapi import FastAPI
from contextlib import asynccontextmanager @asynccontextmanager
async def lifespan(app: FastAPI):
# 启动时初始化Redis连接池
app.state.redis = await create_redis_pool()
yield
# 关闭时释放资源
await app.state.redis.close() app = FastAPI(lifespan=lifespan)

2.2 全局状态管理

from fastapi import FastAPI, Request

app = FastAPI()

@app.middleware("http")
async def add_process_time_header(request: Request, call_next):
start_time = time.time()
response = await call_next(request)
process_time = time.time() - start_time
# 记录到全局状态
request.app.state.request_count += 1
return response

3. 综合应用案例:电商系统架构

from fastapi import APIRouter, Depends
from pydantic import BaseModel class ProductCreate(BaseModel):
name: str
price: float
stock: int router = APIRouter(prefix="/products") @router.post("")
async def create_product(
product_data: ProductCreate,
db: AsyncSession = Depends(get_db),
redis=Depends(get_redis)
):
# 检查商品名称重复
existing = await db.execute(
select(Product).filter(Product.name == product_data.name)
)
if existing.scalar():
raise HTTPException(400, "Product name exists") # 写入数据库
new_product = Product(**product_data.dict())
db.add(new_product)
await db.commit() # 更新缓存
await redis.delete("product_list") return {"id": new_product.id}

课后Quiz

Q1:当遇到数据库连接池耗尽问题时,应该如何排查?

A. 检查数据库服务器状态

B. 增加连接池最大连接数

C. 检查是否忘记释放会话

D. 所有以上选项

正确答案:D。连接池问题需要综合排查,包括服务器资源、配置参数和代码逻辑。

Q2:为什么推荐使用yield方式管理数据库会话?

A. 实现事务的自动提交

B. 确保异常时回滚事务

C. 自动关闭会话连接

D. 所有以上选项

正确答案:D。yield语法可以完美实现会话的生命周期管理。

常见报错解决方案

错误1:RuntimeError: No response returned.

原因:依赖项中未正确返回响应

解决:

async def auth_dependency():
try:
# 验证逻辑
yield
except Exception as e:
return JSONResponse(status_code=401, content={"error": str(e)})

错误2:sqlalchemy.exc.InterfaceError: Connection closed unexpectedly

原因:数据库连接超时

预防:

engine = create_async_engine(
DATABASE_URL,
pool_size=20,
max_overflow=10,
pool_timeout=30
)

错误3:pydantic.error_wrappers.ValidationError

原因:请求体数据验证失败

排查步骤:

  1. 检查请求头Content-Type是否正确
  2. 验证请求体JSON格式
  3. 检查Pydantic模型定义
  4. 使用curl测试请求:
curl -X POST http://localhost:8000/items \
-H "Content-Type: application/json" \
-d '{"name":"example", "price": 9.99}'

余下文章内容请点击跳转至 个人博客页面 或者 扫码关注或者微信搜一搜:编程智域 前端至全栈交流与成长,阅读完整的文章:FastAPI依赖注入与上下文管理 | cmdragon's Blog

往期文章归档:

FastAPI依赖注入与上下文管理的更多相关文章

  1. DI容器Ninject在管理接口和实现、基类和派生类并实现依赖注入方面的实例

    当一个类依赖于另一个具体类的时候,这样很容易形成两者间的"强耦合"关系.我们通常根据具体类抽象出一个接口,然后让类来依赖这个接口,这样就形成了"松耦合"关系,有 ...

  2. ABP(现代ASP.NET样板开发框架)系列之6、ABP依赖注入

    点这里进入ABP系列文章总目录 基于DDD的现代ASP.NET开发框架--ABP系列之6.ABP依赖注入 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)” ...

  3. ABP理论学习之依赖注入

    返回总目录 本篇目录 什么是依赖注入 传统方式产生的问题 解决办法 依赖注入框架 ABP中的依赖注入基础设施 注册 解析 其他 ASP.NET MVC和ASP.NET Web API集成 最后提示 什 ...

  4. ABP依赖注入

    ABP依赖注入 点这里进入ABP系列文章总目录 基于DDD的现代ASP.NET开发框架--ABP系列之6.ABP依赖注入 ABP是“ASP.NET Boilerplate Project (ASP.N ...

  5. ABP中的依赖注入思想

    在充分理解整个ABP系统架构之前首先必须充分了解ABP中最重要的依赖注入思想,在后面会具体举出一些实例来帮助你充分了解ABP中的依赖注入思想,在了解这个之前我们首先来看看什么是依赖注入?来看看维基百科 ...

  6. Orchard详解--第三篇 依赖注入之基础设施

    Orchard提供了依赖注入机制,并且框架的实现也离不开依赖注入如模块管理.日志.事件等.在前一篇中提到在Global.asax中定义的HostInitialization创建了Autofac的IoC ...

  7. ABP官方文档翻译 2.1 依赖注入

    依赖注入 什么是依赖注入 传统方式的问题 解决方案 构造函数注入模式 属性注入模式 依赖注入框架 ABP依赖注入基础设施 注册依赖注入 传统注册 帮助接口 自定义/直接注册 使用IocManager ...

  8. ABP+AdminLTE+Bootstrap Table权限管理系统第四节--仓储,服务,服务接口及依赖注入

    在ABP框架中,仓储,服务,这块算是最为重要一块之一了.ABP框架提供了创建和组装模块的基础,一个模块能够依赖于另一个模块,一个程序集可看成一个模块, 一个模块可以通过一个类来定义这个模块,而给定义这 ...

  9. PHP 依赖注入,依赖反转 (IOC-DI)

    https://my.oschina.net/u/3529405/blog/1821744 <?php /** * 依赖注入 IOC DI * 参考文章 * https://segmentfau ...

  10. 依赖注入(DI)在PHP中的实现

    什么是依赖注入? IOC:英文全称:Inversion of Control,中文名称:控制反转,它还有个名字叫依赖注入(Dependency Injection,简称DI). 当一个类的实例需要另一 ...

随机推荐

  1. 【Netty】(3)—源码NioEventLoopGroup

    netty(3)-源码NioEventLoopGroup 一.概念 NioEventLoopGroup对象可以理解为一个线程池,内部维护了一组线程,每个线程负责处理多个Channel上的事件,而一个C ...

  2. labuladong的二分法查找模板

    几条规则: 1. while(left <= right)作为循环进入条件,退出则为left > right 循环内不再有return条件 2. nums[mid] == target之后 ...

  3. 修改NuGet包默认存放位置

    默认情况下,NuGet下载的包存放在系统盘(C盘中),这样一来,时间长了下载的包越多,C盘占用的空间也就越多. 1.问题描述 默认情况下,NuGet下载的包存放在系统盘(C盘中,一般在路径C:\Use ...

  4. nc命令-Netcat (网络刀)

    https://blog.csdn.net/freeking101/article/details/53289198 nc参数 1) -l 用于指定nc将处于侦听模式.指定该参数,则意味着nc被当作s ...

  5. Linux mint安装外部软件

    //更新系统  sudo apt-get update更新ruanjian sudo apt-get upgrade更新系统 //安装浏览器 sudo apt-get install chromium ...

  6. FLink16--计数窗口--CountWindiwApp

    一.依赖 https://www.cnblogs.com/robots2/p/16048648.html 二.代码 概念:窗口来多少条计算一次,存在滚动和滑动两种情况 package net.xdcl ...

  7. [AI/GPT] 硅基流动(SiliconFlow) : AI大模型时代的基础设施

    概述:硅基流动(SilliconFlow) 简介 硅基流动(SiliconFlow) 是一家专注于人工智能(AI)基础设施的公司,致力于通过技术创新降低大模型(如生成式AI和大语言模型)的部署和推理成 ...

  8. 安全稳定地远程访问飞牛NAS

    春节前从一个网友那里了解到一个新的NAS--飞牛. 起因是我们一个用户用我们的SD-WAN来远程访问飞牛NAS,市面上做NAS的很多,之所以单独体验这个产品主要是: 不需要购买硬件,这个是非常重要的, ...

  9. 科研界DeepSeek+AI应用协作攻略来了!

    自从DeepSeek爆火 AI应用届开启"精英集结" 与DeepSeek携手撑起国产AI一片天 比如,DeepSeek+Midjourney 成为设计师的好帮手 DeepSeek+ ...

  10. Mybatis中的 switch

    我这遇到个问题,如果 type字段为null则查询type is null,否则查对应的值 询问 AI 得知,可以用choose-when-otherwise <select> selec ...