title: FastAPI依赖注入实践:工厂模式与实例复用的优化策略

date: 2025/04/06 01:22:25

updated: 2025/04/06 01:22:25

author: cmdragon

excerpt:

FastAPI依赖注入系统中,类依赖的默认行为是为每个请求创建新实例,可能导致性能问题。通过工厂模式控制实例创建过程,可解耦配置和服务实例化,支持依赖层级嵌套,符合单一职责原则。使用lru_cache实现带缓存的工厂模式,优化高频调用场景性能。单例模式实现真正的单例依赖,请求级别复用策略在请求处理周期内复用实例。实际应用场景包括配置中心集成和多租户系统,动态配置加载和租户感知的依赖注入。常见报错解决方案涉及422 Validation Error和依赖项初始化失败。

categories:

  • 后端开发
  • FastAPI

tags:

  • FastAPI
  • 依赖注入
  • 工厂模式
  • 实例复用
  • 单例模式
  • 多租户系统
  • 性能优化


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

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

FastAPI依赖注入深度实践:类依赖的工厂模式与实例复用

一、类依赖的基本原理

在FastAPI的依赖注入系统中,类作为依赖项使用时,框架会自动创建类的实例。当我们这样定义一个路由处理函数时:

@app.get("/items/")
def read_items(service: ItemService = Depends()):
return service.get_items()

FastAPI会为每个请求创建一个新的ItemService实例。这种默认行为在某些场景下可能产生性能问题,特别是当依赖类需要执行初始化数据库连接、加载大文件等耗时操作时。

二、工厂模式实现

2.1 工厂函数基础实现

通过工厂模式控制实例创建过程:

class DatabaseConfig:
def __init__(self, url: str = "sqlite:///test.db"):
self.url = url class DatabaseService:
def __init__(self, config: DatabaseConfig):
self.connection = self.create_connection(config.url) def create_connection(self, url):
# 模拟数据库连接
print(f"Creating new connection to {url}")
return f"Connection_{id(self)}" def get_db_service(config: DatabaseConfig = Depends()) -> DatabaseService:
return DatabaseService(config) @app.get("/users/")
def get_users(service: DatabaseService = Depends(get_db_service)):
return {"connection": service.connection}

这个实现的特点:

  • 解耦配置和服务的实例化
  • 支持依赖层级嵌套(DatabaseConfig自动注入到工厂函数)
  • 符合单一职责原则

2.2 带缓存的工厂模式

优化高频调用场景的性能:

from fastapi import Depends
from functools import lru_cache class AnalysisService:
def __init__(self, config: dict):
self.model = self.load_ai_model(config["model_path"]) def load_ai_model(self, path):
print(f"Loading AI model from {path}")
return f"Model_{id(self)}" @lru_cache(maxsize=1)
def get_analysis_service(config: dict = {"model_path": "models/v1"}) -> AnalysisService:
return AnalysisService(config) @app.get("/predict")
def make_prediction(service: AnalysisService = Depends(get_analysis_service)):
return {"model": service.model}

缓存机制说明:

  • 使用lru_cache实现内存缓存
  • maxsize=1表示只缓存最新实例
  • 当配置参数变化时会自动创建新实例
  • 适合模型加载等重量级初始化场景

三、实例复用策略

3.1 单例模式实现

实现真正的单例依赖:

from contextlib import contextmanager
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker class DatabaseSingleton:
_instance = None def __new__(cls, dsn: str):
if not cls._instance:
cls._instance = super().__new__(cls)
cls._instance.engine = create_engine(dsn)
cls._instance.Session = sessionmaker(bind=cls._instance.engine)
return cls._instance @contextmanager
def get_db_session(dsn: str = "sqlite:///test.db"):
db = DatabaseSingleton(dsn)
session = db.Session()
try:
yield session
session.commit()
except Exception as e:
session.rollback()
raise e
finally:
session.close() @app.get("/transactions")
def get_transactions(session=Depends(get_db_session)):
return {"status": "success"}

3.2 请求级别复用

在请求处理周期内复用实例:

from fastapi import Request

class RequestTracker:
def __init__(self, request: Request):
self.request = request
self.start_time = time.time() @property
def duration(self):
return time.time() - self.start_time def get_tracker(request: Request) -> RequestTracker:
if not hasattr(request.state, "tracker"):
request.state.tracker = RequestTracker(request)
return request.state.tracker @app.get("/status")
def get_status(tracker: RequestTracker = Depends(get_tracker)):
return {"duration": tracker.duration}

四、实际应用场景

4.1 配置中心集成

动态配置加载示例:

from pydantic import BaseSettings

class AppSettings(BaseSettings):
env: str = "dev"
api_version: str = "v1" class Config:
env_file = ".env" def config_factory() -> AppSettings:
return AppSettings() def get_http_client(settings: AppSettings = Depends(config_factory)):
timeout = 30 if settings.env == "prod" else 100
return httpx.Client(timeout=timeout)

4.2 多租户系统

租户感知的依赖注入:

class TenantContext:
def __init__(self, tenant_id: str):
self.tenant_id = tenant_id
self.config = self.load_tenant_config() def load_tenant_config(self):
# 模拟从数据库加载配置
return {
"db_url": f"sqlite:///tenant_{self.tenant_id}.db",
"theme": "dark" if self.tenant_id == "acme" else "light"
} def tenant_factory(tenant_id: str = Header(...)) -> TenantContext:
return TenantContext(tenant_id) @app.get("/dashboard")
def get_dashboard(ctx: TenantContext = Depends(tenant_factory)):
return {"theme": ctx.config["theme"]}

五、课后Quiz

  1. 工厂模式在依赖注入中的主要作用是?

    A) 减少代码量

    B) 控制实例创建过程

    C) 提高路由处理速度

    D) 自动生成API文档

  2. 使用lru_cache装饰器缓存服务实例时,当什么情况下会创建新实例?

    A) 每次请求时

    B) 输入参数变化时

    C) 服务类代码修改时

    D) 服务器重启时

  3. 在多租户系统中,如何实现不同租户的数据库隔离?

    A) 使用不同的路由前缀

    B) 基于租户ID动态生成数据库连接

    C) 为每个租户创建独立应用实例

    D) 使用请求头认证

(答案:1.B 2.B 3.B)

六、常见报错解决方案

错误1:422 Validation Error

现象

{
"detail": [
{
"loc": [
"header",
"x-tenant-id"
],
"msg": "field required",
"type": "value_error.missing"
}
]
}

原因分析

  • 请求缺少必要的Header参数
  • 工厂函数参数类型声明错误
  • 依赖项层级结构不匹配

解决方案

  1. 检查请求是否包含所有必需的Header
  2. 验证工厂函数的参数类型声明
  3. 使用依赖关系图工具调试:
    uvicorn main:app --reload --debug

错误2:依赖项初始化失败

现象

RuntimeError: Unable to initialize service - missing config

排查步骤

  1. 检查依赖项的参数传递链路
  2. 验证配置对象的默认值设置
  3. 在工厂函数中添加调试日志:
    def get_service(config: AppSettings):
    print("Current config:", config.dict())
    return MyService(config)

预防建议

  • 为所有配置参数设置合理的默认值
  • 使用pydantic的Field验证:
    class AppSettings(BaseSettings):
    db_url: str = Field(..., env="DATABASE_URL")

余下文章内容请点击跳转至 个人博客页面 或者 扫码关注或者微信搜一搜:编程智域 前端至全栈交流与成长,阅读完整的文章:FastAPI依赖注入实践:工厂模式与实例复用的优化策略 | cmdragon's Blog

往期文章归档:

FastAPI依赖注入实践:工厂模式与实例复用的优化策略的更多相关文章

  1. [ASP.NET Core 3框架揭秘] 依赖注入:IoC模式

    原文:[ASP.NET Core 3框架揭秘] 依赖注入:IoC模式 正如我们在<依赖注入:控制反转>提到过的,很多人将IoC理解为一种“面向对象的设计模式”,实际上IoC不仅与面向对象没 ...

  2. ASP.NET Core 6框架揭秘实例演示[05]:依赖注入基本编程模式

    毫不夸张地说,整个ASP.NET Core就是建立在依赖注入框架之上的.ASP.NET Core应用在启动时构建管道所需的服务,以及管道处理请求使用到的服务,均来源于依赖注入容器.依赖注入容器不仅为A ...

  3. .NET CORE 依赖注入 实践总结

    知识点回顾 依赖包. Microsoft.Extensions.DependencyInjection.Abstractions 核心对象和方法. IServiceCollection.注入对象的容器 ...

  4. java 抽象工厂模式简单实例

    抽象工厂模式:提供一个创建一系列的相关的或者依赖的对象的接口,无需指定它们的具体实现类,具体的时间分别在子类工厂中产生. 类似于工厂模式:隔离了具体类的生产实现,使得替换具体的工厂实现类很容易.包含有 ...

  5. laravel服务容器(IOC控制反转,DI依赖注入),服务提供者,门脸模式

    laravel的核心思想: 服务容器: 容器:就是装东西的,laravel就是一个个的对象 放入:叫绑定 拿出:解析 使用容器的目的:这里面讲到的是IOC控制反转,主要是靠第三方来处理具体依赖关系的解 ...

  6. php工厂模式的实例

    * 单例模式:用于创建单一类型的唯一实例对象 * 工厂模式:用于创建多种类型的多个实例对象 //声明形状类 class Shape { //声明静态方法create,根据容器形状不同,创建不同图形类的 ...

  7. C#工厂模式代码实例

    此处示例为一个简易计算器工厂模式的实现. 创建类库,名为CalcLib,我把计算功能全部放在这个类库中. 首先,创建一个抽象的计算器算法父类,如下: /// <summary> /// 计 ...

  8. Java简单工厂模式以及来自lambda的优化

    前言    设计模式是软件工程中一些问题的统一解决方案的模型,它的出现是为了解决一些普遍存在的,却不能被语言特性直接解决的问题,随着软件工程的发展,设计模式也会不断的进行更新,本文介绍的是经典设计模式 ...

  9. php设计模式之简单工厂模式代码实例

    <?php header("Content-type:text/html;charset=utf-8"); /** * 共同接口 */ interface db { func ...

  10. 用工厂模式解决ASP.NET Core中依赖注入的一个烦恼

    这是最近在实际开发中遇到的一个问题,用 asp.net core 开发一个后端 web api ,根据指定的 key 清除 2 台 memcached 服务器上的缓存.背景是我们在进行 .net co ...

随机推荐

  1. 双指针习题:Binary Deque

    14.Binary Deque 题面翻译 Binary Deque - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 有多组数据. 每组数据给出 \(n\) 个数,每个数为 \(0\) ...

  2. Excel函数公式大全(图文详解)

    ---------------------------- ----------------------------------------------------------------------- ...

  3. 玩转云端 | 拥有HBlock这项“存储盘活绝技”,数据中心也能“热辣瘦身”!

    夏天马上就要到了,"瘦身"不光是特定人群的需求,也是数据中心的需求.构建轻量化.低碳化.高性价比的新型数据中心,更有效地支撑经济社会数字化转型,已成为业界主流趋势. 如何让数据中心 ...

  4. Windows中安装和配置Maven

    1.下载 下载地址:https://maven.apache.org/download.cgi 下载文件:https://dlcdn.apache.org/maven/maven-3/3.9.6/bi ...

  5. Q:如何在Linux系统中查看实时网卡流量

    Linux查看实时网卡流量的几种方式 来源  https://www.jianshu.com/p/b9e942f3682c 在工作中,我们经常需要查看服务器的实时网卡流量.通常,我们会通过这几种方式查 ...

  6. lxl 讲课的记录

    D1 lxl:LCT 没有前途.所以平衡树一般只需要 fhq-treap. 线段树.平衡树简单例题 P3215 注意到抵消掉合法括号串之后一定是这样的情况:))))((((即前缀最小值 \(a\).后 ...

  7. Java后台获取微信小程序用户信息、openid

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...

  8. 洛谷P4390 [BalkanOI2007] Mokia 摩基亚 题解

    题目传送门. 想必 我的另外一篇题解 已经把这道题的思路说的很清楚了,但是那道题是把所有的修改全部告诉你,然后再一个一个问你矩阵和,但是这道题他是修改中夹着询问,但是没有关系,我们照样可做. 考虑将所 ...

  9. tinyint、int的区别

    1.tinyint(1字节--4位[带符号]) 很小的整数.带符号的范围是-128到127.无符号的范围是0到255. 2.smallint(2字节--6位[带符号]) 小的整数.带符号的范围是-32 ...

  10. 原生开发,使用C语言调用Windows API 开发软件思路分享

    Githu: https://github.com/vladelaina/Catime 作者是一个高度依赖计时器功能的人,但是市面上的软件都不能满足个性化的需求,所以打算自己动手开发,同时采用c语言来 ...