现象
最近在调试接口的时候,发现fastjson输出到前端页面的字符串带了一些奇怪的字符,比如$ref
分析原因
这其实是由于对象的重复引用导致的。因为fastjson在做序列化的时候,会把已经序列化过的对象存到一个map中保存起来。所以在每次进行序列化的时候都会去缓冲查找是否已经序列化过该对象,如果发现有序列化过该对象,就不会进行序列化了,而是返回它的一个引用 $ref。
引用标识
“$ref”:”..” 上一级
“$ref”:”@” 当前对象,也就是自引用
“$ref”:”$” 根对象
{"$ref":"../.."} 引用父对象的父对象
“$ref”:”$.children.0” 基于路径的引用,相当于root.getChildren().get(0)
public boolean writeReference(JSONSerializer serializer, Object object, int fieldFeatures) {
SerialContext context = serializer.context;
int mask = SerializerFeature.DisableCircularReferenceDetect.mask;
if (context == null || (context.features & mask) != 0 || (fieldFeatures & mask) != 0) {
return false;
}
if (serializer.references != null && serializer.references.containsKey(object)) {
serializer.writeReference(object);
return true;
} else {
return false;
}
}
案例复现
import com.alibaba.fastjson.JSONObject;
public static void main(String[] args) {
Car car1 = new Car(999,"bwm");
List<Car> list1 =new ArrayList();
list1.add(car1);
JSONObject cars = new JSONObject();
cars.put("list1",list1);
cars.put("list2",list1);
cars.put("list3",list1);
System.out.println(cars.toJSONString());
}
Connected to the target VM, address: '127.0.0.1:56012', transport: 'socket'
{"list1":[{"brand":"bwm","price":999}],"list3":[{"$ref":"$.list1[0]"}],"list2":[{"$ref":"$.list1[0]"}]}
Disconnected from the target VM, address: '127.0.0.1:56012', transport: 'socket'
Process finished with exit code 0
解决方法
1,当jsonObject的value是一个集合时,不要重复添加集合中的源码,这会导致一个重复引用问题;所以建议在重复引用时,对对象采用深拷贝,把拷贝对象添加到不同的集合中;
2,在构建jsonobject的时候,可以禁用探测重复引用的功能,但这可能会导致一个严重的问题,overstack;
|