Python使用FastAPI实现文件上传接口的最佳方案

 更新时间:2025年11月09日 08:42:07   作者:Java私教  
在现代 Web 应用中,文件上传几乎是所有系统的标配功能:无论是头像上传、文档提交,还是批量导入数据, 本文将详细介绍如何使用 FastAPI 快速、高效地实现文件上传接口,需要的朋友可以参考下

引言

在现代 Web 应用中,文件上传几乎是所有系统的标配功能:无论是头像上传、文档提交,还是批量导入数据。 本文将详细介绍如何使用 FastAPI 快速、高效地实现文件上传接口,并给出进阶场景(如多文件上传、限制文件大小、异步保存文件等)的最佳实践。

一、FastAPI 文件上传的核心机制

FastAPI 内置了强大的文件上传支持,基于 starletteUploadFile 对象实现。 它的底层使用 Python 的异步文件处理机制,性能优于传统的同步文件读取。

在 FastAPI 中,文件上传参数可以通过以下两种类型声明:

  • File(...):用于声明表单中的文件字段。
  • UploadFile:封装上传文件对象,包含 filenamecontent_typefile 等属性。

二、单文件上传示例

先看一个最简单的例子:上传一个文件并保存到服务器。

from fastapi import FastAPI, File, UploadFile
import shutil
import os

app = FastAPI()

UPLOAD_DIR = "uploads"
os.makedirs(UPLOAD_DIR, exist_ok=True)

@app.post("/upload/")
async def upload_file(file: UploadFile = File(...)):
    file_path = os.path.join(UPLOAD_DIR, file.filename)
    
    # 保存文件
    with open(file_path, "wb") as buffer:
        shutil.copyfileobj(file.file, buffer)
    
    return {"filename": file.filename, "content_type": file.content_type}

说明:

  • UploadFile 是一个异步文件对象(底层为 SpooledTemporaryFile)。
  • file.file 是标准的类文件对象,可直接读写。
  • shutil.copyfileobj() 用于高效复制二进制数据流。

访问接口:

curl -F "file=@example.png" http://127.0.0.1:8000/upload/

返回:

{
  "filename": "example.png",
  "content_type": "image/png"
}

三、多文件上传

FastAPI 支持一次性上传多个文件,只需将类型改为 List[UploadFile]

from typing import List
from fastapi import UploadFile, File

@app.post("/upload/multiple/")
async def upload_multiple(files: List[UploadFile] = File(...)):
    saved_files = []
    for file in files:
        file_path = os.path.join(UPLOAD_DIR, file.filename)
        with open(file_path, "wb") as buffer:
            shutil.copyfileobj(file.file, buffer)
        saved_files.append(file.filename)
    return {"uploaded_files": saved_files}

上传多个文件:

curl -F "files=@a.png" -F "files=@b.jpg" http://127.0.0.1:8000/upload/multiple/

四、异步写入与性能优化

上面的写入方式是同步的。如果文件较大,可能阻塞事件循环。 可以使用异步文件写入方式提升性能(推荐在高并发或大文件场景):

import aiofiles

@app.post("/upload/async/")
async def upload_file_async(file: UploadFile = File(...)):
    file_path = os.path.join(UPLOAD_DIR, file.filename)
    async with aiofiles.open(file_path, 'wb') as out_file:
        while content := await file.read(1024):
            await out_file.write(content)
    await file.close()
    return {"filename": file.filename}

提示:aiofiles 是一个异步文件操作库,可与 FastAPI 无缝协作。

安装依赖:

pip install aiofiles

五、限制上传文件大小与类型

限制大小(Nginx / Uvicorn 层)

FastAPI 本身不限制文件大小。建议在 网关层(如 Nginx)限制上传大小:

client_max_body_size 10M;

校验文件类型

你可以在接口中检查文件类型:

ALLOWED_TYPES = ["image/png", "image/jpeg"]

@app.post("/upload/image/")
async def upload_image(file: UploadFile = File(...)):
    if file.content_type not in ALLOWED_TYPES:
        return {"error": "Unsupported file type"}
    # 继续保存逻辑...
    return {"filename": file.filename}

六、结合 Pydantic 校验表单 + 文件混合上传

在实际业务中,文件上传通常伴随表单数据(如用户ID、描述等)。 FastAPI 支持文件与表单字段混合处理:

from fastapi import Form

@app.post("/upload/with_form/")
async def upload_with_form(
    user_id: int = Form(...),
    description: str = Form(""),
    file: UploadFile = File(...)
):
    file_path = os.path.join(UPLOAD_DIR, file.filename)
    with open(file_path, "wb") as buffer:
        shutil.copyfileobj(file.file, buffer)
    return {"user_id": user_id, "description": description, "filename": file.filename}

七、实际应用场景建议

场景推荐方案
上传头像、小文件使用 UploadFile + 异步保存
批量导入Excel、CSV使用 aiofiles 逐行读取,解析后入库
上传到云存储(如OSS、S3)直接使用 boto3oss2 SDK,将文件流转存至云端
接口网关安全限制使用 Nginx / Traefik 限制文件大小和MIME类型
文件命名冲突使用 UUID 或时间戳重命名文件

八、总结

FastAPI 让文件上传的实现变得异常简单且高效。 通过本文的实践,你已经学会了:

  • ✅ 基本文件上传与保存
  • ✅ 多文件与异步写入
  • ✅ 文件类型与大小校验
  • ✅ 表单与文件混合上传
  • ✅ 性能与安全最佳实践

接下来,你可以尝试:

  • 实现一个图片上传 + 自动压缩接口
  • 结合 JWT 用户认证,构建安全的文件上传系统
  • 将文件元数据存储在数据库中,实现可追踪的文件管理系统

建议

在实际项目中,推荐将文件上传逻辑封装为独立的服务模块(如 services/file_service.py), 结合日志(如 loguru)和异常处理中间件,让系统更加可维护与可追踪。

以上就是Python使用FastAPI实现文件上传接口的最佳方案的详细内容,更多关于Python FastAPI文件上传接口的资料请关注脚本之家其它相关文章!

相关文章

  • python基础学习之生成器与文件系统知识总结

    python基础学习之生成器与文件系统知识总结

    本文是参考《python数据分析》的附录对生成器和文件系统结合案例的一个简单回顾,文中对python生成器与文件系统作了非常详细的介绍,对正在学习python的小伙伴们有很好地帮助,需要的朋友可以参考下
    2021-05-05
  • Pycharm关于远程JupyterLab以及JupyterHub登录问题

    Pycharm关于远程JupyterLab以及JupyterHub登录问题

    这篇文章主要介绍了Pycharm关于远程JupyterLab以及JupyterHub登录问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-06-06
  • Django--权限Permissions的例子

    Django--权限Permissions的例子

    今天小编就为大家分享一篇Django--权限Permissions的例子,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-08-08
  • python可视化之颜色映射详解

    python可视化之颜色映射详解

    Python的可视化有很多种,这篇文章主要介绍了Python可视化的颜色映射,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-09-09
  • python sklearn数据预处理之数据缩放详解

    python sklearn数据预处理之数据缩放详解

    数据的预处理是数据分析,或者机器学习训练前的重要步骤,这篇文章主要为大家详细介绍了sklearn数据预处理中数据缩放的相关知识,感兴趣的小伙伴可以学习一下
    2023-10-10
  • pytorch 自定义卷积核进行卷积操作方式

    pytorch 自定义卷积核进行卷积操作方式

    今天小编就为大家分享一篇pytorch 自定义卷积核进行卷积操作方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-12-12
  • python学习之列表的运用

    python学习之列表的运用

    这篇文章主要介绍了python学习之列表的运用,文章首先通过创建列表展开列表运用的相关资料,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-05-05
  • Python 图片文字识别的实现之PaddleOCR

    Python 图片文字识别的实现之PaddleOCR

    OCR方向的工程师,之前一定听说过PaddleOCR这个项目,其主要推荐的PP-OCR算法更是被国内外企业开发者广泛应用,短短半年时间,累计Star数量已超过15k,频频登上Github Trending和Paperswithcode 日榜月榜第一
    2021-11-11
  • python小球落地问题及解决(递归函数)

    python小球落地问题及解决(递归函数)

    这篇文章主要介绍了python小球落地问题及解决(递归函数),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-05-05
  • python实现密码强度校验

    python实现密码强度校验

    这篇文章主要为大家详细介绍了python实现密码强度校验,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03

最新评论