使用fastjson时出现$ref: "$.list[0]"的解决办法(重复引用)

什么是重复/循环引用

简单说,重复引用就是一个集合/对象中的多个元素/属性同时引用同一对象,循环引用就是集合/对象中的多个元素/属性存在相互引用导致循环。

举例说明

  1. 重复引用

    List<Object> list = new ArrayList<>();
    Object obj = new Object();
    list.add(obj);
    list.add(obj);

     

  2. 循环引用

    // 循环引用的特殊情况,自引用
    Map<String,Object> map = new HashMap<>();
    map.put("map",map);
    //
    // map1引用了map2,而map2又引用map1,导致循环引用
    Map<String,Object> map1 = new HashMap<>();
    Map<String,Object> map2 = new HashMap<>();
    map1.put("map",map2);
    map2.put("map",map1);

     

循环引用会触发的问题

暂时不说重复引用,单说循环引用。
一般来说,存在循环引用问题的集合/对象在序列化时(比如Json化),如果不加以处理,会触发StackOverflowError异常。

分析原因:

当序列化引擎解析map1时,它发现这个对象持有一个map2的引用,转而去解析map2。解析map2时,发现他又持有map1的引用,又转回map1。如此产生StackOverflowError异常。

FastJson对重复/循环引用的处理

首先,fastjson作为一款序列化引擎,不可避免的会遇到循环引用的问题,为了避免StackOverflowError异常,fastjson会对引用进行检测。

如果检测到存在重复/循环引用的情况,fastjson默认会以“引用标识”代替同一对象,而非继续循环解析导致StackOverflowError。

以上文两例说明,查看json化后的输出

  1. 重复引用 JSON.toJSONString(list)

[
    {},	 //obj的实体
    {
        "$ref": "$[0]"	 //对obj的重复引用的处理
    }
]

 


     2.循环引用 JSON.toJSONString(map1)

{
// map1的key:value对
    "map": {
    	 // map2的key:value对
        "map": {
        	 // 指向map1,对循环引用的处理
            "$ref": ".."
        }
    }
}
 

引用标识说明:

“$ref”:”..” 上一级
“$ref”:”@” 当前对象,也就是自引用
“$ref”:”$” 根对象
“$ref”:”$.children.0” 基于路径的引用,相当于root.getChildren().get(0)

关闭FastJson的引用检测

1
JSON.toJSONString(object, SerializerFeature.DisableCircularReferenceDetect);

FastJson提供了SerializerFeature.DisableCircularReferenceDetect这个序列化选项,用来关闭引用检测。关闭引用检测后,重复引用对象时就不会被$ref代替,但是在循环引用时也会导致StackOverflowError异常。

避免重复引用序列化时显示$ref

  1. 在编码时,使用新对象为集合或对象赋值,而非使用同一对象
    不要在多处引用同一个对象,这可以说是一种java编码规范,需要时刻注意。
  2. 不要关闭FastJson的引用检测来避免显示$ref
    引用检测是FastJson提供的一种避免运行时异常的优良机制,如果为了避免在重复引用时显示$ref而关闭它,会有很大可能导致循环引用时发生StackOverflowError异常。这也是FastJson默认开启引用检测的原因。

避免重复/循环引用的正确姿势

  1. 重复引用

    1
    2
    3
    4
    5
    6
    7
    8
    
    List<Object> list = new ArrayList<>();
    Object obj = new Object();
    list.add(obj);
    // 创建新的对象
    Object newObj = new Object();
    // 使用org.springframework.beans.BeansUtils复制属性值
    BeansUtils.copy(obj, newObj);
    list.add(obj);
    
  2. 循环引用
    循环引用这种逻辑本身就不合理,需要在编码时注意避免,这是逻辑错误而非编码技巧。

 

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要解决 "$ref": "$.value.data[0].node" 错误,你可以尝试以下几个步骤: 1. 检查 JSON 数据的结构:确保 JSON 数据中包含了名为 "value" 的键,并且其值是一个对象。在该对象中,确保存在一个名为 "data" 的键,其值是一个数组,并且该数组中至少有一个元素。每个元素应该是一个包含 "node" 键的对象。 2. 使用合适的 JSON 库:Java 有多个流行的 JSON 库可供选择,如 Jackson、Gson、Fastjson 等。确保你使用的库支持 JSON Pointer 表达式,并且在解析 JSON 数据正确处理该表达式。 3. 配置 JSON 库以支持 JSON Pointer:有些 JSON 库可能需要进行额外的配置才能正确处理 JSON Pointer 表达式。查阅相关库的文档以了解如何配置和使用 JSON Pointer。 4. 执行 JSON Pointer 表达式:使用库提供的方法,根据指定的 JSON Pointer 表达式获取到所需的值。对于 "$.value.data[0].node" 这个表达式,你可以使用库提供的相应方法来获取值。 以下是使用 Jackson 库处理 JSON 数据并解析 JSON Pointer 表达式的示例代码: ```java import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; import com.github.fge.jsonpatch.JsonPatch; import com.github.fge.jsonpatch.JsonPatchException; // 假设你的 JSON 数据保存在一个字符串中 String jsonString = "{ \"value\": { \"data\": [ { \"node\": \"value1\" } ] } }"; // 创建 ObjectMapper 对象 ObjectMapper objectMapper = new ObjectMapper(); // 解析 JSON 数据 JsonNode jsonNode = objectMapper.readTree(jsonString); // 使用 JSON Pointer 表达式获取值 JsonNode nodeValue = jsonNode.at("/value/data/0/node"); String value = nodeValue.asText(); System.out.println(value); // 输出: value1 ``` 请根据你使用JSON 库进行相应的调整。如果问题仍然存在,可以提供更多的代码和错误信息以便进一步帮助你解决问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值