enum的json反序列化问题 415 Unsupported Media Type

简介: enum的反序列过程中如果使用了jackson的注解,很可能会产生一个神奇的bug

enum的反序列化问题 415 Unsupported Media Type

当你的程序在线上出现了error code 415,而本地环境正常的时候,可能就是由于enum的反序列化导致的。

当在enum的反序列化工厂方法上添加了@JsonCreator,且该方法有多个参数,每个参数前都加了@JsonProperty注解,但没有在JsonProperty中显式的指定value为参数名时,就会导致这一问题。

这是一个神奇的bug,在本地测试的时候,不管是用postman还是前端页面发送参数,后端的controller都可以正常的接收,将json的字符串,反序列化为实体类。但是一到了线上,就显示415 content type not supported

{

   "timestamp": "2021-08-13T07:29:27.173+0000",

   "status": 415,

   "error": "Unsupported Media Type",

   "message": "Content type 'application/json;charset=UTF-8' not supported",

   "path": "/ops/submit_inventory_evaluate"

}

经检查restcontroller requestmapping requestbody注解都放在了合适位置,且前端js里content type也采用了application/json,或postman在header里content type也是如此设置。最初怀疑为前端的问题,反复检查无误后,通过postman测试,发现还是415错误,之后将问题定位到了后端

因为本地环境正常,导致测试很困难,只能一点点的排查,发到线上环境再测试,最终将问题锁定到了实体类中一个enum属性。在加了@JsonIgnore注解后,线上环境就可以正常的反序列化了,在查看enum中的toEnum方法时,注意到了上面的@JsonCreator注解,仔细阅读jackson2.9.0中关于该注解的说明。发现该注解当作用于有多个参数的构造或工厂方法上时,每个参数都需要加@JsonProperty,同时在该注解中需要显式的指定value的值,对应哪个属性。除非你使用了支持检测参数名称的扩展模块,因为JDK8之前,默认 JDK 版本无法字节码中存储或检索参数名称。

通过这段话,我开始猜测是本地Oracle的JDK和线上JDK的区别,但是都是JDK8应该都支持这一特性。后面通过搜索这一特性——反射获取方法的参数名称,发现除了要求JDK8及以上,还有一点是编译的时候必须指定编译选项:-parameters,来打开这一特性的支持,默认是关闭的状态。

打开这一特性有三种方法:

  1. 手动命令方式编译:javac -parameters XXX.java


  1. IDE(以Idea为例)编译:Settings -> Build,Execution,Deployment -> Compiler -> Java Compiler


  • Additional command line parameters框中设置为-parameters
  • Override compiler parameters per-module中新增一个module并设置 Compilation options值为-parameters
  • image.png
  1. Maven编译:通过编译插件指定,保证项目迁移的正确性(推荐)

其中第二种方法是IDEA,默认开启的,通过删除这一参数设置,测试发现本地也出现了415错误,最终确认了问题的来源。

目录
相关文章
|
2月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
196 1
|
2月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
223 1
|
3月前
|
XML JSON 编解码
从JSON到Protobuf,深入序列化方案的选型与原理
序列化是数据跨边界传输的“翻译官”,将结构化数据转为二进制流。JSON可读性强但冗余大,Protobuf高效紧凑、性能优越,成主流选择。不同场景需权衡标准化与定制优化,选最合适方案。
317 3
|
5月前
|
JSON 人工智能 Go
在Golang中序列化JSON字符串的教程
在Golang中,使用`json.Marshal()`可将数据结构序列化为JSON格式。若直接对JSON字符串进行序列化,会因转义字符导致错误。解决方案包括使用`[]byte`或`json.RawMessage()`来避免双引号被转义,从而正确实现JSON的序列化与反序列化。
254 7
|
6月前
|
XML JSON Java
go语言之JSON序列化
本文介绍了Go语言中的JSON序列化与反序列化,其操作与Java类似。需要注意的是,由于Go语言的包管理机制,变量和引入包的首字母需大写,以便其他包引用。示例代码展示了如何将`Student`结构体进行JSON序列化(返回字节数组,需转为字符串)及反序列化。此外,文章还说明了通过tag(如`json`和`xml`)指定序列化变量的重要性,以避免因包间访问限制导致反序列化失败或值为null的问题。
128 0
|
XML 存储 JSON
Twaver-HTML5基础学习(19)数据容器(2)_数据序列化_XML、Json
本文介绍了Twaver HTML5中的数据序列化,包括XML和JSON格式的序列化与反序列化方法。文章通过示例代码展示了如何将DataBox中的数据序列化为XML和JSON字符串,以及如何从这些字符串中反序列化数据,重建DataBox中的对象。此外,还提到了用户自定义属性的序列化注册方法。
209 1
|
7月前
|
JSON JavaScript 前端开发
Go语言JSON 序列化与反序列化 -《Go语言实战指南》
本文介绍了 Go 语言中使用 `encoding/json` 包实现 JSON 与数据结构之间的转换。内容涵盖序列化(`Marshal`)和反序列化(`Unmarshal`),包括基本示例、结构体字段标签的使用、控制字段行为的标签(如 `omitempty` 和 `-`)、处理 `map` 和切片、嵌套结构体序列化、反序列化未知结构(使用 `map[string]interface{}`)以及 JSON 数组的解析。最后通过表格总结了序列化与反序列化的方法及类型要求,帮助开发者快速掌握 JSON 数据处理技巧。
|
JSON JavaScript 前端开发
Go语言中json序列化的一个小坑,建议多留意一下
在Go语言开发中,JSON因其简洁和广泛的兼容性而常用于数据交换,但其在处理数字类型时存在精度问题。本文探讨了JSON序列化的一些局限性,并介绍了两种替代方案:Go特有的gob二进制协议,以及msgpack,两者都能有效解决类型保持和性能优化的问题。
394 7
|
JSON 前端开发 JavaScript
聊聊 Go 语言中的 JSON 序列化与 js 前端交互类型失真问题
在Web开发中,后端与前端的数据交换常使用JSON格式,但JavaScript的数字类型仅能安全处理-2^53到2^53间的整数,超出此范围会导致精度丢失。本文通过Go语言的`encoding/json`包,介绍如何通过将大整数以字符串形式序列化和反序列化,有效解决这一问题,确保前后端数据交换的准确性。
349 4
|
JSON JavaScript Java
对比JSON和Hessian2的序列化格式
通过以上对比分析,希望能够帮助开发者在不同场景下选择最适合的序列化格式,提高系统的整体性能和可维护性。
427 3