博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
使用fastjson时出现$ref: "$.list[2]"的解决办法(重复引用)
阅读量:6258 次
发布时间:2019-06-22

本文共 4502 字,大约阅读时间需要 15 分钟。

hot3.png

最近下做成绩分析的功能的时候,出现了$ref: "$.list[2]"的情况。

然后我查了一下出现这种情况的原因和解决办法。

出现$ref: "$.list[2]"的原因是因为循环引用/内存对象重复

那么什么是重复/循环引用?

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

举例说明

重复引用

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

循环引用

// 循环引用的特殊情况,自引用  Map
map = new HashMap<>(); map.put("map",map); // // map1引用了map2,而map2又引用map1,导致循环引用 Map
map1 = new HashMap<>(); Map
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":"../.."} 引用父对象的父对象
“$ref”:”$.children.0” 基于路径的引用,相当于root.getChildren().get(0)

解决方法:

知道出现的原因后,我发现我写那个接口是确实有重复引用的,因为需求要有个最低科的详细情况,还有挂科的各个科详细情况,所以当某个学生挂科了,那么他最低科和挂科的那个list里面的对象就重复了。当时这个需求是这样,也只能寻找其他方法了。

关闭FastJson的引用检测

查看一下fastjson的API,有两种关闭方法。

局部的

JSON.toJSONString(object, SerializerFeature.DisableCircularReferenceDetect);

全局的

普通的spring项目的话,用xml配置

text/html;charset=UTF-8
WriteMapNullValue
WriteNullStringAsEmpty
DisableCircularReferenceDetect

如果springboot的话

public class FastJsonHttpMessageConverterEx extends FastJsonHttpMessageConverter{    public FastJsonHttpMessageConverterEx(){        //在这里配置fastjson特性(全局设置的)        FastJsonConfig fastJsonConfig = new FastJsonConfig();        //fastJsonConfig.setDateFormat("yyyy-MM-dd HH:mm:ss");    //自定义时间格式        //fastJsonConfig.setSerializerFeatures(SerializerFeature.WriteMapNullValue);  //正常转换null值        //fastJsonConfig.setSerializerFeatures(SerializerFeature.DisableCircularReferenceDetect);     //关闭循环引用        this.setFastJsonConfig(fastJsonConfig);    }    @Override    protected boolean supports(Class
clazz) { return super.supports(clazz); }}@Configurationpublic class WebMvcConfigurer extends WebMvcConfigurerAdapter { ..... @Bean public FastJsonHttpMessageConverterEx fastJsonHttpMessageConverterEx(){ return new FastJsonHttpMessageConverterEx(); }}

配置这个DisableCircularReferenceDetect的作用是:决定了生成的“多个”JSON对象中,是否加载被引用的同一个对象的数据。

开启和关闭FastJson的“循环引用检测”特性的对比

be876025ac4197294983898c72d76797c4a.jpg

现在的问题:

全局关闭引用检测确实是方便,但是有可能的以后的代码越来越多的情况出现StackOverflowError。但是局部的关闭引用检测的话也有些小问题。JSON.toJSONString(object, SerializerFeature.DisableCircularReferenceDetect) 这个方法是把对象转为String对象,如果在@ResponseBody注解的方法里返回这个String对象的话,就会像我之前那篇文章所说那些样

在加了@ResponseBody注解的Controller中使用

String result = JSON.toJSONString(obj);return result;

这种情况就相当于JSON.toJSONString() 这句话执行了两次。

当然你也可以HttpServletResponse 输出流的方法输出这个字符串,不然为了代码风格统一,还是不采用这种。

有没有更好的解决方法?最终还回到那个原因上,因为我们重复引用同一个对象,所以现在解决方法应该是把那个挂科成绩对象score 转到另外一个新的对象。

List list = new ArrayList<>();Object obj = new Object();list.add(obj);// 创建新的对象Object newObj = new Object();// 使用org.springframework.beans.BeansUtils复制属性值BeansUtils.copyProperties(obj, newObj);list.add(newObj);

BeanUtils可以通过反射机制将两个对象进行属性的拷贝,但是他们不是指向同一个地址,相当好用,假如你的bean有很多个属性,你就不用逐个复制属性了。

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

在编码时,使用新对象为集合或对象赋值,而非使用同一对象

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

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

1、重复引用

可以向上面的代码一样,创建新的对象,把要有重复需要的对象的属性复制给新对象,新对象再添加到json那里。
2、循环引用
循环引用这种逻辑本身就不合理,需要在编码时注意避免,这是逻辑错误而非编码技巧。

作者:carway
链接:https://www.jianshu.com/p/6041242405e8
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处.

转载于:https://my.oschina.net/zjllovecode/blog/2251632

你可能感兴趣的文章
从观察者模式到手写EventEmitter源码
查看>>
当你在浏览器里输入一个url发生了什么?
查看>>
React入门0x007: 生命周期概念
查看>>
webpack调优总结
查看>>
Springboot对多线程的支持详解
查看>>
Sublime text3修改tab键为缩进四个空格
查看>>
「Do.008」Android 实战项目(3)——Git 分支管理模型
查看>>
原生js中Object.keys方法详解
查看>>
Webpack 4.X 从入门到精通 - plugin(二)
查看>>
Elasticsearch的搜索类型(SearchType类型)
查看>>
Java知识点总结(JDBC-大文本对象的使用)
查看>>
javascript 正则命名分组
查看>>
以太坊开发实战学习-solidity语法 (三)
查看>>
Windows Theano GPU 版配置
查看>>
vue2.0学习笔记(九):vue项目实战--持续更新(1)
查看>>
Vue.js入门教程-过滤器
查看>>
Python之使用Pandas库实现MySQL数据库的读写
查看>>
基于scikit-learn机器学习库的分类预测
查看>>
svg与视频结合的镂空效果实践总结
查看>>
Scikit中的特征选择,XGboost进行回归预测,模型优化的实战
查看>>