fastjson转化入参字符串的特殊字段名$ref问题

本文探讨了Fastjson在处理JSON对象时遇到的 `$ref` 引用问题,如何通过Feature.DisableCircularReferenceDetect特性解决转化逻辑冲突,并提及不同Fastjson版本对这一规则的影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、问题描述

{
    "schema": {
        "originalRef": "Result«object»",
        "$ref": "#/definitions/Result«object»"
    }
}

入参字符串是json格式,可直接转为JSONObject,由于某个字段是$ref,导致调用fastjson的JSON.parseObject(String text)方法出现问题。
1、问题一:$ref在第一个字段,转化api报错:
com.alibaba.fastjson.JSONException: syntax error
2、问题二:$ref在第二个字段,则转化api未报错,但是同级所有字段全部消失,只留下$ref这个字段,且值变为@:
在这里插入图片描述
注意:若返回转化后的json给前端,则会报错:Could not write JSON: Infinite recursion (StackOverflowError); nested exception is com.fasterxml.jackson.databind.JsonMappingException: Infinite recursion (StackOverflowError) (through reference chain: 统一返回响应类路径)

二、解决思路

从报错情况和fastjson使用经验判断,$ref是fastjson声明引用属性名,相当于内置字段,外来字段重名会导致转化逻辑。所以考虑处理"引用"这个问题。

三、解决方案

使用转化规则DisableCircularReferenceDetect,即关闭引用。最终代码:

JSON.parseObject(text, Feature.DisableCircularReferenceDetect);

四、小结

1、使用$ref作为入参字段会影响fastjson转化逻辑,需要使用Feature.DisableCircularReferenceDetect转化规则处理;
2、JSON.parseObject(text, Feature.DisableCircularReferenceDetect),转化规则在1.2.24版本并未生效,1.2.39+版本才生效,1.2.24-1.2.39之间的版本暂未测试。

FastJSON 的 `toJSONString` 方法是一个用于序列化对象的核心功能之一。它能够将 Java 对象换为 JSON 字符串表示形式,同时支持多种自定义配置选项来满足不同的需求。 以下是关于 FastJSON 中 `toJSONString` 方法及其引用处理的一些关键点: ### 1. 序列化过程中的循环引用检测 当存在复杂的对象图结构时,可能会遇到循环引用的情况(即 A 引用了 B,而 B 又反向引用了 A)。为了防止无限递归导致栈溢出错误,FastJSON 提供了一个内置机制来检测并解决这种问题。通过设置 `SerializerFeature.DisableCircularReferenceDetect` 或启用默认的循环引用保护模式[^6],可以控制如何处理这些情况。 ```java // 启用循环引用检测,默认情况下会抛出异常 JSONObject.toJSONString(object, SerializerFeature.DisableCircularReferenceDetect); ``` 如果未禁用此特性,则会在发现循环引用时自动将其替换为字符串 `"ref"` 加上对应的内存地址标识[^7]。 ### 2. 自定义引用管理 除了简单的循环引用外,在某些场景下可能还需要更精细地控制哪些字段应该被标记为引用或者忽略掉不必要的重复数据传输。这可以通过实现 `SerializeFilter` 接口来自定义逻辑[^8]: ```java public class CustomRefFilter implements SerializeFilter { public JSONObject process(Object object, String name, Object value) { if (value instanceof SomeSpecificType && alreadySerialized.contains(value)) { return new Reference(value.hashCode()); } return null; } private Set<Object> alreadySerialized = new HashSet<>(); } ``` 随后可以在调用 `toJSONString()` 方法时传入该过滤器实例: ```java CustomRefFilter filter = new CustomRefFilter(); String jsonString = JSON.toJSONString(yourObject, filter); ``` 上述代码片段展示了如何创建一个定制化的引用处理器,并应用于目标实体类之上[^9]。 ### 3. YAML 配置与全局设定 对于 Spring Boot 项目而言,还可以利用 application.yml 文件来进行一些高级别的调整,比如修改默认的行为方式等。例如下面的例子就说明了怎样通过外部化属性文件指定特定版本号以及开启其他优化开关[^4]: ```yaml spring: json: pretty-print: true date-format: yyyy-MM-dd HH:mm:ss disable-circular-reference-detect: false ``` 注意这里的路径仅为示意用途实际应用需依据具体框架文档确认最终键名格式[^10]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值