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

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

【需求与环境描述】

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  
### FastJSON 序列化概述 FastJSON 是阿里巴巴开源的一个高性能 JSON 序列化和反序列化工具包。其设计目标是提供简单易用且性能卓越的功能,适用于多种应用场景。 #### 使用方法 要使用 FastJSON 进行对象的序列化与反序列化,首先需要引入相应的 Maven 或 Gradle 依赖项: 对于 Maven 用户来说,在 `pom.xml` 文件中添加如下配置: ```xml <dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson</artifactId> <version>最新版本号</version> </dependency> ``` 而对于 Gradle 用户,则可以在 build.gradle 中加入这行代码: ```groovy implementation 'com.alibaba:fastjson:最新版本号' ``` 完成依赖导入之后就可以开始编写具体的业务逻辑了。下面给出几个简单的例子展示如何利用 FastJSON 实现基本功能[^1]。 #### 基本示例 ##### 将 Java 对象转换为 JSON 字符串 (序列化) 假设有一个名为 Person 的实体类,可以通过以下方式将其实例转化为 JSON 格式的字符串表示形式: ```java import com.alibaba.fastjson.JSON; public class Main { public static void main(String[] args) { Person person = new Person(); person.setName("张三"); person.setAge(20); String jsonString = JSON.toJSONString(person); // 转换为 json string System.out.println(jsonString); } } ``` ##### 把 JSON 字符串解析回 Java 对象 (反序列化) 如果已经有了一个描述某个对象状态的 JSON 字符串,那么可以很容易地通过 FastJSON 解析得到对应的 Java Bean 实例: ```java import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.TypeReference; // ... TypeReference<Person> typeRef = new TypeReference<Person>() {}; Person parsedObject = JSON.parseObject("{\"name\":\"李四\",\"age\":25}", typeRef); System.out.println(parsedObject.getName() + ", " + parsedObject.getAge()); ``` #### 常见问题及其解决方案 - **日期格式处理不当** 当涉及到 Date 类型字段时,默认情况下可能会遇到时间戳不一致等问题。为了确保正确性和一致性,建议设置全局属性或针对特定场景指定日期模式: ```java // 设置默认日期格式 DateFormatUtils.setDefaultFormat("yyyy-MM-dd HH:mm:ss"); // 或者在每次调用前临时设定 JSON.defaultTimeZone(TimeZone.getTimeZone("GMT+8")); JSON.defaultDateFormat("yyyy-MM-dd HH:mm:ss"); ``` - **循环引用引发异常** 某些复杂的数据结构可能存在父子关系或其他类型的环形链表情况,这时如果不做特殊处理就会抛出 StackOverflowError 错误。为此,FastJSON 提供了一个非常有用的特性——自动检测并忽略这些危险连接: ```java // 启用安全模式防止无限递归 SerializeConfig config = SerializeConfig.globalInstance; config.put(Person.class, new SimplePropertyPreFilter()); // 或者直接传入参数控制行为 JSON.toJSONString(object, SerializerFeature.WriteMapNullValue, SerializerFeature.DisableCircularReferenceDetect); ``` - **敏感信息泄露风险** 考虑到安全性因素,在实际开发过程中应当注意保护用户的隐私资料免受不必要的暴露。一种做法是在导出之前先过滤掉那些不应该公开显示的内容;另一种则是采用加密算法对重要部分进行编码后再传输给客户端。 ```java // 定义白名单只保留必要的键值对 Set<String> includes = Sets.newHashSet("id", "username"); SimplePropertyPreFilter filter = new SimplePropertyPreFilter(includes.toArray(new String[includes.size()])); // 执行序列化操作的同时应用上述规则 String safeJsonStr = JSON.toJSONString(userProfile, filter); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值