赞
踩
在构建一个公共的批处理方法类的时候,在测试输出的时候,打印了" r e f " : " ref":" ref":"[0][0]"的内容,这让我比较疑惑。不由得继续了下去…
首先,我们需要明确 ,在使用诸如Java的序列化库(如Jackson、Gson或Fastjson等)将数据转换为JSON字符串时,JSON.toJSONString(map<String,String>) 调用中可能出现 “ r e f " : " ref":" ref":"[0][0]” 的原因。在JSON序列化过程中,“ r e f " : " ref":" ref":"[0][0]” 这类引用标记通常表示对象中存在循环引用,即一个对象直接或间接地引用了自己。JSON序列化库在检测到这种循环引用时,会尝试使用引用来避免无限递归,并节省内存。
对于简单的 Map<String, String> 类型,通常不应该出现循环引用,因为键值对本身不包含对其他键值对的引用。因此,这个问题可能是在其他部分的代码或序列化库的实现中产生的。
假设我们有一个简单场景,其中Map中的值是一个自引用的类实例,这会导致序列化时出现$ref。
import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.annotation.JsonIdentityInfo; import com.fasterxml.jackson.annotation.ObjectIdGenerators; @JsonIdentityInfo( generator = ObjectIdGenerators.PropertyGenerator.class, property = "id") class SelfReferencingObject { int id; SelfReferencingObject selfRef; public SelfReferencingObject(int id) { this.id = id; } public void setSelfRef(SelfReferencingObject ref) { this.selfRef = ref; } } public class Demo { public static void main(String[] args) throws Exception { ObjectMapper mapper = new ObjectMapper(); SelfReferencingObject obj1 = new SelfReferencingObject(1); SelfReferencingObject obj2 = new SelfReferencingObject(2); // 创建循环引用 obj1.setSelfRef(obj1); obj2.setSelfRef(obj2); Map<String, SelfReferencingObject> map = new HashMap<>(); map.put("obj1", obj1); map.put("obj2", obj2); String json = mapper.writeValueAsString(map); System.out.println(json); } }
上述代码在运行时,将会输出类似于以下内容,其中包含了$ref来表示循环引用:
{
“obj1”: {
“id”: 1,
“@ref”: “KaTeX parse error: Expected 'EOF', got '}' at position 8: [0]" }̲, "obj2": { …[1]”
}
}
如果Map中的值直接或间接地引用了Map本身或其他位于Map中的对象,形成了一个闭环,序列化时为了防止无限循环和堆栈溢出,序列化库会使用$ref来标记已处理过的对象,避免重复输出。
即使没有循环引用,但如果多个键值对引用了相同的对象实例,一些序列化库也会使用$ref来优化输出,表示这些位置引用的是同一个对象。
大多数序列化库提供了配置选项来禁用循环引用检测,但这可能会导致其他问题,如无限循环序列化。
ObjectMapper mapper = new ObjectMapper();
mapper.disable(com.fasterxml.jackson.databind.deser.DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
mapper.enable(com.fasterxml.jackson.databind.SerializationFeature.WRITE_DURABLE_OBJECT_IDS);
String json = mapper.writeValueAsString(map);
...
注意:此方法可能不会直接解决问题,且可能导致无限循环或其它错误,实际应用中应谨慎。
你可以通过实现自定义的序列化器或采用库提供的注解等方式,控制特定对象或类的序列化行为,避免$ref的产生。
// 使用Jackson的@JsonIdentityInfo注解解决循环引用
// 上面的SelfReferencingObject类已经添加了@JsonIdentityInfo注解
// 序列化代码保持不变
使用@JsonIdentityInfo后,输出的JSON会为重复的对象生成唯一ID,而不是直接使用$ref。
在序列化前,检查并打破潜在的循环引用,比如将引用替换为ID或者浅拷贝对象以切断循环链。
public class Demo { public static void main(String[] args) throws Exception { ObjectMapper mapper = new ObjectMapper(); SelfReferencingObject obj1 = new SelfReferencingObject(1); SelfReferencingObject obj2 = new SelfReferencingObject(2); // 防止循环引用,这里不设置selfRef Map<String, SelfReferencingObject> map = new HashMap<>(); map.put("obj1", obj1); map.put("obj2", obj2); String json = mapper.writeValueAsString(map); System.out.println(json); } }
这个例子中,我们直接不设置selfRef,从而避免了循环引用。
某些库如Jackson提供了@JsonIdentityInfo注解来处理循环引用问题,自动为重复的对象生成ID引用。
已在示例1中展示了如何使用Jackson的@JsonIdentityInfo注解处理循环引用。
或者使用JSON.toJSONString(Object object, SerializerFeature… features)方法,并传入SerializerFeature.DisableCircularReferenceDetect特性来禁用循环引用检测。
String jsonString = JSON.toJSONString(users, SerializerFeature.DisableCircularReferenceDetect);
在Java中,对象之间的循环引用、重复引用、自引用或相互引用,通常在代码层面直观体现为对象的属性互相指向对方。下面通过示例来具体展示这些引用方式:
当两个或多个对象互相持有对方的引用,形成一个闭环,这就是循环引用。
class Person { String name; Person friend; Person(String name) { this.name = name; } void setFriend(Person friend) { this.friend = friend; // 这里设置朋友的friend为自己,形成循环引用 friend.setFriend(this); } } public class Main { public static void main(String[] args) { Person alice = new Person("Alice"); Person bob = new Person("Bob"); alice.setFriend(bob); // 设置Alice的朋友是Bob } }
在这个例子中,alice的朋友是bob,而bob的朋友又被设置为alice,形成了循环引用。
如果多个变量或数据结构引用同一个对象实例,就是重复引用。
class Book { String title; Book(String title) { this.title = title; } } public class Main { public static void main(String[] args) { Book popularBook = new Book("Popular Title"); List<Book> library = new ArrayList<>(); library.add(popularBook); library.add(popularBook); // 同一个Book实例被添加两次,形成重复引用 } }
这里,popularBook这个Book实例被library列表重复引用了两次。
自引用指的是对象的一个属性直接或间接地引用自身。
class Node { String data; Node next; // 可能指向自己,形成自引用 Node(String data) { this.data = data; } void setNext(Node next) { this.next = next; } } public class Main { public static void main(String[] args) { Node node = new Node("Node Data"); // 形成自引用 node.setNext(node); } }
在Node类的例子中,next属性可以设置为指向自己,形成自引用。
在解决这个问题时,关键是要找到循环引用的来源。这可能需要你深入检查代码和序列化库的实现。一旦找到循环引用的来源,你就可以采取适当的措施来避免它,例如修改代码逻辑或自定义序列化过程。如果问题是由序列化库引起的,更新到最新版本或寻找替代库可能是一个解决方案。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。