我使用flexjson
序列化列表。序列化后,我打印对象hashcode()
,我会检查对象是否相等。
问题,操作后对象引用正在改变。我不想要它。
我的代码;
String sameOfString="sameOfString";
List<Object> list=new ArrayList<Object>();
list.add(sameOfString);
list.add(sameOfString);
list.add("differentString");
for(Object str:list){
System.out.println(str.hashCode());
}
**System.out.println(list.get(0)==list.get(1));**
System.out.println(list.get(0)==list.get(2));
System.out.println("--");
String json= new JSONSerializer().exclude("*.class").deepSerialize(list);
List<String> listDeserilized = (List<String>) new JSONDeserializer().deserialize(json);
for(Object str:listDeserilized){
System.out.println(str.hashCode());
}
**System.out.println(listDeserilized.get(0)==listDeserilized.get(1));**
System.out.println(listDeserilized.get(0)==listDeserilized.get(2));
输出..
1018069468
1018069468
199675306
真
假
-
1018069468
1018069468
199675306
假
假
如何解决这个问题?
答案 0 :(得分:0)
如何解决这个问题?
你没有。反序列化对象意味着在内存中的不同位置创建新实例。
但是,我不推荐这个,您可以保留对象的注册表(可能是HashMap
),执行反序列化,检查注册表是否包含该对象,以及用现有的替换它。
String
类已经在实习中为您提供了一些此类行为。请参阅String#intern()
。
答案 1 :(得分:0)
你正在努力的是平等是什么意思?在Java中,有两个版本的相等(与许多现代语言一样)。
这分别是.equals()和==之间的区别。当您反序列化某些内容时,它会创建一个新对象,因此如果您有另一个该对象的副本,则会将其序列化。它永远不会是2号相等(即通过引用相等)。因为反序列化被赋予了JSON表示,并且它从中创建了一个新的对象图。
但它可能是.equals(),前提是你的对象正确定义了.equals()对于包含的所有对象的含义。默认情况下,Object.equals()比较引用,与==相同。但是,每个对象都可以重载它,并且在String.equals()的情况下比较字符串的实际内容。
所以只需更改代码即可使用.equals()而不是==并获得您的期望:
System.out.println( list.get(0).equals( list.get(1) ) )
System.out.println( list.get(0).equals( list.get(2) ) )
答案 2 :(得分:0)
我用jackson lib解决了问题。
@JsonIdentityInfo
注释保护对象关系(换句话说;保护引用)
在反序列化之后,对象位置在内存中不会改变。
最终代码:
package com.maar.projectbuilder;
import com.fasterxml.jackson.annotation.JsonIdentityInfo;
import com.fasterxml.jackson.annotation.JsonTypeInfo;
import com.fasterxml.jackson.annotation.JsonTypeName;
import com.fasterxml.jackson.annotation.ObjectIdGenerators;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
/**
* Created by Firat on 11.9.2014.
*/
public class Testt {
@JsonDeserialize(as=StringClazz.class)
@JsonTypeInfo(use=JsonTypeInfo.Id.CLASS, include=JsonTypeInfo.As.PROPERTY, property="@class")
@JsonIdentityInfo(generator=ObjectIdGenerators.IntSequenceGenerator.class, property="@id")
@JsonTypeName(value = "StringClazz")
private static class StringClazz{
private String data="";
public StringClazz() {
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
}
public static void main(String[] args) throws IOException {
StringClazz someOfString=new StringClazz();
someOfString.setData("someOfString");
StringClazz differentString=new StringClazz();
differentString.setData("differentString");
List<StringClazz> list=new ArrayList<StringClazz>();
list.add(someOfString);
list.add(someOfString);
list.add(differentString);
for(Object str:list){
System.out.println(str.hashCode());
}
System.out.println(list.get(0)==list.get(1));
System.out.println(list.get(0)==list.get(2));
System.out.println("-------");
String json= null;
try {
json = new ObjectMapper().enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL).writeValueAsString(list);
} catch (JsonProcessingException e) {
e.printStackTrace();
}
System.out.println(json);
List<StringClazz> listDeserilized =new ObjectMapper().enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL).readValue(json, new TypeReference<List<StringClazz>>() {
});
for(Object str:listDeserilized){
System.out.println(str.hashCode());
}
System.out.println(listDeserilized.get(0)==listDeserilized.get(1));
System.out.println(listDeserilized.get(0)==listDeserilized.get(2));
}
}