0

0

Pydantic 中如何精准校验 JSON 可序列化字典类型?

聖光之護

聖光之護

发布时间:2026-02-19 10:50:20

|

468人浏览过

|

来源于php中文网

原创

Pydantic 中如何精准校验 JSON 可序列化字典类型?

本文介绍 Pydantic 内置的 JsonValue 类型,用于严格校验符合 JSON 规范的嵌套数据结构(如字典、列表、基本类型),避免使用宽松类型(如 Dict[str, Any])导致非 JSON 值(如 datetime、set、自定义对象)意外通过验证。

本文介绍 pydantic 内置的 `jsonvalue` 类型,用于严格校验符合 json 规范的嵌套数据结构(如字典、列表、基本类型),避免使用宽松类型(如 `dict[str, any]`)导致非 json 值(如 `datetime`、`set`、自定义对象)意外通过验证。

在构建 API 或处理外部输入(如 Webhook、配置文件、前端 JSON payload)时,常需确保某字段是真正可被 JSON 序列化的字典——即其键为字符串,值仅限于 str、int、float、bool、None、list 或 dict,且所有嵌套层级均满足该约束。若误用 Dict[str, Any],虽能接受任意 Python 对象,却可能隐含不可序列化的值(如 datetime.now()、Decimal('3.14') 或自定义类实例),导致后续 json.dumps() 报错,破坏系统健壮性。

Pydantic 从 v2.5 起正式提供 JsonValue 类型(位于 pydantic.types.JsonValue),专为此场景设计。它是一个递归联合类型,精确对应 JSON 的语法规范:

from pydantic.types import JsonValue
from pydantic import BaseModel, ValidationError

class MyModel(BaseModel):
    config: JsonValue  # ✅ 支持任意合法 JSON 结构:{}、[]、"str"、123、true、null 等

JsonValue 的完整定义等价于:

JsonValue = Union[
    List["JsonValue"],
    Dict[str, "JsonValue"],
    str,
    bool,
    int,
    float,
    None,
]

⚠️ 注意:JsonValue 不包含 bytes、decimal.Decimal、datetime、UUID、Enum 等非原生 JSON 类型;也不允许 tuple(会被转为 list,但类型校验失败)、set 或自定义对象。

AI抖音
AI抖音

AI抖音,会思考的抖音

下载

实际校验示例

# 正确输入(全部通过)
valid_cases = [
    {"name": "Alice", "scores": [95.5, 87], "active": True},
    {"meta": None, "tags": ["v1", "prod"]},
    42,
    "hello",
    [1, "two", {"nested": True}],
]

for data in valid_cases:
    try:
        model = MyModel(config=data)
        print(f"✅ Valid: {data}")
    except ValidationError as e:
        print(f"❌ Invalid: {data} → {e}")

# 错误输入(全部抛出 ValidationError)
invalid_cases = [
    {"created_at": datetime.now()},     # datetime 不可 JSON 序列化
    {"value": decimal.Decimal("3.14")}, # Decimal 不在 JsonValue 范围内
    {"items": (1, 2, 3)},               # tuple 不被允许(即使内容合法)
    {"data": {b"key": "value"}},         # bytes key 非 str
]

for data in invalid_cases:
    try:
        MyModel(config=data)
    except ValidationError as e:
        print(f"❌ Rejected: {data} → {e.errors()[0]['msg']}")

若仅需“JSON 字典”(非任意 JSON 值)

如业务明确要求字段必须是 dict(而非 str/int 等其他 JSON 类型),可组合使用:

from typing import Dict
from pydantic.types import JsonValue

class StrictDictModel(BaseModel):
    payload: Dict[str, JsonValue]  # ✅ 必须是 dict,且每个 value 满足 JSON 规范

这比 Dict[str, Any] 更安全,也比手动递归定义 Json = Union[...] 更简洁可靠(后者易因前向引用或循环定义引发 RecursionError 或 IDE 类型推导失败)。

总结与最佳实践

  • 首选 JsonValue:当字段需承载任意合法 JSON 数据(对象、数组、原子值)时,直接使用 JsonValue,语义清晰且开箱即用。
  • 组合 Dict[str, JsonValue]:当业务强约束为“必须是字典”时,显式声明键类型与值类型边界。
  • 避免 Dict[str, Any]:除非你主动承担序列化风险并自行后处理,否则会丧失类型安全与早期错误拦截能力。
  • ? 注意版本兼容性:JsonValue 自 Pydantic v2.5 引入,确保 pydantic>=2.5.0;旧版本可考虑 pydantic.json_schema.JsonSchemaValue(非运行时校验)或自定义验证器,但推荐升级。

通过 JsonValue,你能在模型层就筑牢 JSON 兼容性防线,让数据契约更明确、错误更早暴露、API 更加可靠。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

442

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

544

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

322

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

81

2025.09.10

css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

591

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

573

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

216

2023.09.04

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

561

2026.02.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 9.4万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.3万人学习

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

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