文档章节

【FastJSON】解决FastJson中“$ref 循环引用”的问题

SuShine
 SuShine
发布于 2016/07/28 17:36
字数 1138
阅读 60
收藏 0

【需求与环境描述】

0、开发环境

SSH,EasyUI,MySQL

1、需求要求:

(1)首先获取所有的贷款订单数据,即List <LoanOrder>。

(2)然后从单个贷款订单实体LoanOrder去访问贷款人实体Loaner的信息。

2、实体之间的关系描述

(1)LoanOrder实体与Loaner实体是双向的多对一和一对多关系。

(2)LoanOrder是“多方”,其中的关系属性为“private Loaner loaner”。

(3)Loaner是“一方”,其中的关系属性为“Set<LoanOrder> orders”。

3、代码示例

(1) 贷款订单LoanOrder代码

@Entity
@Table(name = "t_bp_loan_order")
public class LoanOrder implements java.io.Serializable {
  
  /*省略其他次要属性*/
  private Loaner loaner;

  @ManyToOne(fetch = FetchType.LAZY)
  @JoinColumn(name = "LOANER_ID")
  public Loaner getLoaner() {
    return this.loaner;
  }
}

(2) Loaner方代码

@Entity
@Table(name = "t_bp_loaner")
public class Loaner implements java.io.Serializable {

  /* 其他普通属性略去 */
  private Set<LoanOrder> loanOrders = new HashSet<LoanOrder>(0);

  @OneToMany(fetch = FetchType.LAZY, mappedBy = "loaner")
  @JSONField(serialize = true)
  public Set<LoanOrder> getLoanOrders() {
    return this.loanOrders;
  }
}

【障碍再现】

1、需求01:

首先获取所有的贷款订单数据,即List <LoanOrder>,发现贷款人“ 张三 ”有两个订单。

服务器后台传送到前台页面上的JSON数据如下所示:

2、需求02:

然后,依次在第一个和第二个贷款订单中点击“张三”,从而去访问“名字叫做‘张三’”贷款人实体Loaner的信息。

结果,第一个订单可以显示贷款人的数据,但是在第二个订单数据中,不能获取到“loaner(贷款人)”的数据,并且loaner中提示“$ref”。

【解决方案】

第一步: 禁用 FastJson的“循环引用检测”特性。

1、核心代码

2、作用

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

在此,决定了生成的“多个”贷款订单JSON对象中,是否加载被引用的同一个贷款人JSON对象的数据。

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

当从服务器端传来的多个LoanOrder对象通过FASTJSON被序列化到“前端”后,会被浏览器解析成“DOM”对象。 

(1)当开启FastJson的“循环引用检测”特性时:

1  对于第一个LoanOrder 01,fastjson会完全解析并加载它的所有数据,包括它所关联的Loaner贷款人信息,如下图所示。

2  对于第二个LoanOrder 02,fastjson仅仅解析并加载其贷款订单部分的数据,对于“$ref”所指向的 Loaner贷款人的数据,fastjson会因为“ 开启了 fastJson 的‘循环引用检测’机制 ”而不去加载该贷款人数据。

当加载第二个贷款订单数据时,fastjson检测到已经在第一个订单LoanOrder 01中加载了“贷款人Loaner”的数据,fastjson会因为“开启了‘循环引用检测’机制”而不去再次加载该贷款人数据,而仅仅将一个指向第一个贷款订单LoanOrder01中“贷款人”的引用赋值给第二个贷款订单中的贷款人的位置。

因此,在生成的第二个贷款订单的JSON串中,对于贷款人信息,仅仅只有一个“ $ref ”。

而jQuery这个前端技术又无法解析该引用,因此,就无法读取贷款人的数据,如下图所示。

第二步:禁止Loaner对象获取Set<LoanOrder>的数据。

方法一: 将原来的“双向关系”修改为“单向关系”

1、原来:LoanOrder与Loaner之间是双向关系。

2、修改后:只能从LoanOrder访问Loaner,从Loaner无法访问到LoanOrder。

3、具体方法01

重要前提:不删除Loaner中的“Set<LoanOrder> orders”属性。

注意,若在采用注解映射实体类的方式中,没有使用“@Transient”注解,则数据库会报错。

4、具体方法 02

直接删除“Set<LoanOrder> orders”属性极其相关的setter()和getter()方法。

方法二: 不修改关系的前提下,禁止序列化

在不修改LoanOrder和Loaner双向关系的情况下,Loaner对象中的Set<LoanOrder>集合完成数据的加载,当其向前端Browser传递JSON数据时,禁止序列化Set<LoanOrder>集合。

具体方法:

设置注解“@JSONField(serialize =false)”。

说明:

A.“@JSONField”是fastjson提供的注解标签,其作用为控制其所标注的属性“能否被序列化”。

B.在此其作用为:禁止"loanOrders"这个Set集合被序列化。

具体如下图所示。

【解决后的效果】

读者如要转载,请标明出处和作者名,谢谢。

地址01:

http://space.itpub.net/25851087

地址02:http://www.cnblogs.com/zjrodger

作者名:zjrodger  

本文转载自:

共有 人打赏支持
SuShine
粉丝 126
博文 565
码字总数 154303
作品 0
朝阳
后端工程师
私信 提问
解决fastjson内存对象引用/循环引用导致json中出现$ref

解决fastjson内存对象引用/循环引用导致json中出现$ref 赵伊凡's Blog2018-01-031 阅读 缓存java技术 目前公司有一部分数据使用的是类似于 redis 的一种自研 数据库 进行存储的。对于一般的对...

赵伊凡's Blog
2018/01/03
0
0
使用fastjson时出现$ref: "$.list[2]"的解决办法(重复引用)

最近下做成绩分析的功能的时候,出现了$ref: "$.list[2]"的情况。 然后我查了一下出现这种情况的原因和解决办法。 出现$ref: "$.list[2]"的原因是因为循环引用/内存对象重复 那么什么是重复/...

小黄狗
2018/10/24
0
0
解决fastjson内存对象重复/循环引用json错误

什么是重复/循环引用 简单说,重复引用就是一个集合/对象中的多个元素/属性同时引用同一对象,循环引用就是集合/对象中的多个元素/属性存在相互引用导致循环。 举例说明 重复引用 List<Objec...

寻知者
2016/12/20
191
0
fastjson发布1.1.3版本

Fastjson 1.1.3是一个重要的功能增强版本,在这个版本开始,fastjson支持循环引用,完善支持WriteClassName的特性,使得fastjson开始具备替换hessian和java serialize的能力。 支持循环引用。...

wenshao
2011/08/29
2.1K
4
使用fastjson中遇到的两个问题$ref

fastjson出现$ref 是由于循环引用引起的,比方说返回的json对象为空,而你要去引用对象的属性,这时候就无法获得了,会出现这样的引用标识,可以这样解决JSONObject.toJSONString(content, ...

writeademo
2016/10/21
81
0

没有更多内容

加载失败,请刷新页面

加载更多

rabbitMQ 在spring 的使用

一、准备工作 maven依赖 <dependency>  <groupId>com.rabbitmq</groupId>  <artifactId>amqp-client</artifactId>  <version>4.0.2</version></dependency> <dependency......

狼王黄师傅
昨天
1
0
Android JNI总结

0x01 JNI介绍 JNI是Java Native Interface的缩写,JNI不是Android专有的东西,它是从Java继承而来,但是在Android中,JNI的作用和重要性大大增强。 JNI在Android中起着连接Java和C/C++层的作...

天王盖地虎626
昨天
1
0
大数据教程(11.8)Hive1.2.2简介&初体验

上一篇文章分析了Hive1.2.2的安装,本节博主将分享Hive的体验&Hive服务端和客户端的使用方法。 一、Hive与hadoop直接的关系 Hive利用HDFS存储数据,利用MapReduce查询数据。 二、Hive与传统数...

em_aaron
昨天
3
0
跟我学Spring Cloud(Finchley版)-15-Hystrix监控详解

Hystrix提供了监控Hystrix Command的能力,本节来详细探讨。 监控端点与数据 应用整合Hystrix,同时应用包含spring-boot-starter-actuator 依赖,就会存在一个/actuator/hystrix.stream 端点...

周立_ITMuch
昨天
7
0
day26:shell题

1、 判断当前主机的CPU生产商,其信息在/proc/cpuinfo文件中vendor id一行中。 如果其生产商为AuthenticAMD,就显示其为AMD公司; 如果其生产商为GenuineIntel,就显示其为Intel公司; 否则,...

芬野de博客
昨天
8
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部