FastAPI WebSocket:揭秘流畅的双向沟通秘诀

文章标题:

FastAPI中WebSocket的运用:探索高效双向交互之道

文章内容:

示例代码运行环境

Python 3.8+

安装依赖:pip install fastapi==0.68.0 uvicorn==0.15.0 websockets==10.3 pydantic==1.10.7

一、WebSocket路由声明规范

1.1 基础路由定义

通过@app.websocket装饰器来定义WebSocket路由:

from fastapi import FastAPI, WebSocket

app = FastAPI()


@app.websocket("/ws/chat/{room_id}")
async def websocket_chat(websocket: WebSocket, room_id: int):
    await websocket.accept()
    try:
        while True:
            data = await websocket.receive_text()
            await websocket.send_text(f"Room {room_id}: {data}")
    except WebSocketDisconnect:
        print("Client disconnected")

关键要点解读:
* 路径参数通过URL直接传递(如{room_id})
* WebSocket对象会自动注入到路由函数中
* 必须显式调用await websocket.accept()来建立连接
* 通过receive_text()send_text()实现双向的数据通信

1.2 路由参数验证

结合路径参数与查询参数进行复合验证:

from fastapi import Query, WebSocket


@app.websocket("/ws/secure/{client_id}")
async def secure_ws(
        websocket: WebSocket,
        client_id: int = Path(..., gt=0),
        token: str = Query(..., min_length=8)
):
    if not validate_token(token):
        await websocket.close(code=1008)
        return
    # ...连接处理逻辑...


## 二、客户端连接建立与握手验证

### 2.1 握手过程控制
自定义握手验证流程:

```python
@app.websocket("/ws/auth")
async def auth_ws(websocket: WebSocket):
    # 手动控制握手过程
    await websocket.accept()

    # 获取握手时的请求头
    headers = websocket.headers
    auth_token = headers.get("authorization")

    if not verify_jwt_token(auth_token):
        await websocket.close(code=1008, reason="Invalid credentials")
        return

    # 验证通过后的处理逻辑

2.2 验证失败处理模式

graph TD
A[客户端请求连接] --> B{验证Headers}
B -- 成功 --> C[返回101状态码]
B -- 失败 --> D[返回403状态码]
C --> E[建立双向通信]
D --> F[关闭TCP连接]

三、连接状态维护与生命周期管理

3.1 连接状态跟踪

使用字典来维护活跃的连接:

from typing import Dict

active_connections: Dict[str, WebSocket] = {}


@app.websocket("/ws/status")
async def status_ws(websocket: WebSocket):
    await websocket.accept()
    client_id = str(websocket.client)
    active_connections[client_id] = websocket

    try:
        while True:
            # 保持连接活跃
            await websocket.receive_text()
    except WebSocketDisconnect:
        del active_connections[client_id]

3.2 心跳检测机制

import asyncio


async def heartbeat(websocket: WebSocket, interval: int = 30):
    try:
        while True:
            await asyncio.sleep(interval)
            await websocket.send_json({
                "type": "heartbeat",
                "timestamp": time.time()
            })
    except WebSocketDisconnect:
        print("Heartbeat terminated")


@app.websocket("/ws/realtime")
async def realtime_ws(websocket: WebSocket):
    await websocket.accept()
    # 启动独立心跳任务
    heartbeat_task = asyncio.create_task(heartbeat(websocket))

    try:
        # 主消息处理循环
        while True:
            data = await websocket.receive_json()
            # 处理业务逻辑...
    finally:
        heartbeat_task.cancel()
        await websocket.close()


## 课后 Quiz

  1. 当客户端发送非文本格式数据时如何处理?

```python
# 正确处理方法
try:
    data = await websocket.receive_json()
except WebSocketDisconnect:
    # 处理断开逻辑
except ValueError:
    await websocket.send_text("ERROR: 仅支持JSON格式")

# 错误处理方法:直接使用未校验的receive()
  1. 如何获取客户端的真实IP地址?
# 正确方法
client_ip = websocket.client.host

# 常见错误:直接读取X-Forwarded-For头
# 需要配合代理设置处理

常见报错解决方案

错误1:403 Handshake Failed

现象 :客户端连接时立即断开
分析 :未通过握手验证,常见于:

  • 缺少必要认证头
  • 验证逻辑返回False后未及时close()
    解决
# 在拒绝连接时立即关闭
if not valid:
    await websocket.close(code=1008)
    return  # 必须立即返回

错误2:1006 Abnormal Closure

现象 :客户端非正常断开
处理方案

try:
    while True:
        data = await websocket.receive_text()
except WebSocketDisconnect as e:
    print(f"断开代码:{e.code}")

错误3:TypeError: 'str' expected

场景 :发送非文本数据时
正确做法

# 发送二进制数据
await websocket.send_bytes(binary_data)

# 发送文本数据
await websocket.send_text("message")

# 发送JSON
await websocket.send_json({"key": "value"})

余下文章内容请点击跳转至 个人博客页面 或者 扫码关注或者微信搜一搜:编程智域 前端至全栈交流与成长
,阅读完整的文章:如何在FastAPI中玩转WebSocket,让实时通信不再烦恼?

往期文章归档:

免费好用的热门在线工具

文章整理自互联网,只做测试使用。发布者:Lomu,转转请注明出处:https://www.it1024doc.com/13051.html

(0)
LomuLomu
上一篇 2025 年 8 月 4 日
下一篇 2025 年 8 月 5 日

相关推荐

  • 工业物联网时序数据库选择攻略:Apache IoTDB缘何成首选

    工业物联网时序数据库选择策略:Apache IoTDB为何成首选 引言:时序数据管理的时代挑战 随着工业4.0和物联网技术的飞速推进,全球范围内的时序数据呈现出井喷式增长态势。据IDC预测,到2025年,全球物联网设备产生的数据量将飙升至79.4ZB,其中超六成属于时序数据。这类数据具备显著特征:高频采集(可达毫秒级)、维度多元(单设备涵盖数百个监测指标)、…

    2025 年 9 月 19 日
    10700
  • pycharm激活码在哪里输入?附pycharm破解步骤

    声明:本文所涉及的 PyCharm 破解补丁与激活码均源自网络公开资源,仅限个人学习研究,禁止任何商业用途。如涉侵权,请联系删除。条件允许时,请支持正版! 先放一张“战果图”:PyCharm 2025.2.1 已顺利激活至 2099 年,爽! 下面用图文一步步演示如何给最新版 PyCharm 打补丁。 嫌折腾?直接买官方正版,全家桶低至 32 元/年,一键登…

    PyCharm激活码 5天前
    1700
  • 2025年最新DataGrip激活码及永久破解教程(支持2099年)

    JetBrains系列工具(包括IDEA、PyCharm、DataGrip等)的破解方法一直备受开发者关注。本文将详细介绍如何获取最新DataGrip激活码并完成永久破解,让你的开发工具有效期延长至2099年! 破解效果预览 先来看看破解成功后的效果截图,可以看到许可证有效期已成功延长至2099年: 本教程适用于所有主流操作系统,无论你使用的是哪个版本的Da…

    2025 年 5 月 10 日
    32000
  • 2025年最新DataGrip永久破解教程(附激活码/注册码)🔥

    还在为DataGrip的试用期烦恼吗?🤔 本教程将手把手教你如何轻松破解DataGrip至2099年!适用于Windows/macOS/Linux系统,支持Jetbrains全家桶(IDEA、PyCharm、Goland等)! 先来看看破解成功的效果图,有效期直接拉到2099年,简直不要太爽!🎉 📥 第一步:下载DataGrip安装包 还没安装的小伙伴看这里…

    DataGrip激活码 2025 年 6 月 28 日
    24000
  • 官方渠道同步最新版pycharm激活码与破解教程

    声明:以下 PyCharm 破解补丁与激活码均搜集自互联网,仅限个人学习,禁止商用。若条件允许,请支持正版!如涉侵权,请联系我删除。 先放张 PyCharm 2025.2.1 成功激活到 2099 年的截图镇楼,爽翻! 下面用图文方式手把手教你搞定最新版 PyCharm 的破解流程。 嫌折腾?直接买官方正版账号,全家桶一键登录,低至 32 元/年:https…

    2025 年 10 月 15 日
    10500

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信