- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用 GSON 将数据保存并恢复到我的应用程序中。问题是,在某些情况下,我有两个不同对象引用的对象,我的意思是,引用了同一个实例。所以多条路径可以通向同一个对象。
当持久化我的模型的解码 GSON 字符串时,如果在两个地方存在同一个对象的两个引用,它持久化它们显然是正确的,但是当再次打开应用程序并加载数据和解码 GSON 字符串时,两个正在创建同一对象的不同实例,而不是同一实例。在第一个实例中进行更改不会反射(reflect)在第二个实例中,因为在解码 json 后是不同的对象。
这是问题的踪迹:
有模型、人和车:
public class Model{
Car car;
Person person;
}
public class Person{
Car car;
}
我将相同的汽车实例设置为模型和人:
Car car = new Car();
model.setCar(car);
person.setCar(car);
car 是 car 和 person 中的同一个实例,现在我用 GSON 对数据进行编码和持久化:
Gson gson = new Gson();
String json = gson.toJson(model);
然后我关闭应用程序并重新打开应用程序,然后解码 json 字符串以恢复模型:
Gson gson = new Gson();
gson.fromJson(json, Model.class);
现在,我有两个不同的 Car 实例,一个在 person 内部,另一个在 Model 内部,它们不是同一个实例,但必须是同一个实例!如果我修改了 Model 的汽车,那么 person 的汽车没有被修改,这是一个错误。
如何解决?
最佳答案
Gson
默认情况下不提供任何方式来缓存实例并检查它是否已经被看到。为此,我们需要实现自定义 com.google.gson.TypeAdapterFactory
。此外,我们需要假设 Car
类(如果需要,还有 Person
类)正确实现了 public boolean equals(Object o)
和 public int hashCode()
所以我们可以使用 Map
来缓存所有实例。
假设您的模型如下所示:
class Model {
private Car car;
private Person person;
// getters, setters, toString
}
class Person {
private int id;
private String name;
private Car car;
// getters, setters, toString
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Person person = (Person) o;
return id == person.id;
}
@Override
public int hashCode() {
return Objects.hash(id);
}
}
class Car {
private int id;
private String name;
// getters, setters, toString
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Car car = (Car) o;
return id == car.id;
}
@Override
public int hashCode() {
return Objects.hash(id);
}
}
Car
和 Person
类有我们用来区分实例的 id
字段。您可以在实现中使用任何您想要的属性。
使用 Map
缓存实例的自定义适配器实现:
class CachedInstancesTypeAdapterFactory implements TypeAdapterFactory {
private final Map<Class, Map> cachedMaps = new HashMap<>();
public CachedInstancesTypeAdapterFactory(Set<Class> customizedClasses) {
Objects.requireNonNull(customizedClasses);
customizedClasses.forEach(clazz -> cachedMaps.compute(clazz, (c, m) -> new HashMap<>()));
}
public final <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
if (cachedMaps.containsKey(type.getRawType())) {
final TypeAdapter<T> delegate = gson.getDelegateAdapter(this, type);
return createCustomTypeAdapter(delegate);
}
return null;
}
@SuppressWarnings("unchecked")
private <T> TypeAdapter<T> createCustomTypeAdapter(TypeAdapter<T> delegate) {
return new TypeAdapter<T>() {
@Override
public void write(JsonWriter out, T value) throws IOException {
delegate.write(out, value);
}
@Override
public T read(JsonReader in) throws IOException {
Object deserialized = delegate.read(in);
Map tInstances = Objects.requireNonNull(cachedMaps.get(deserialized.getClass()));
return (T) tInstances.computeIfAbsent(deserialized, k -> deserialized);
}
};
}
}
下面是如何使用它的示例:
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.TypeAdapter;
import com.google.gson.TypeAdapterFactory;
import com.google.gson.reflect.TypeToken;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonWriter;
import java.io.IOException;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
public class GsonApp {
public static void main(String[] args) {
Gson gson = createGson();
String json = gson.toJson(createModel());
System.out.println(json);
Model result = gson.fromJson(json, Model.class);
System.out.println(result);
System.out.println("Two car instances are the same: " + (result.getCar() == result.getPerson().getCar()));
}
private static Model createModel() {
Car car = new Car();
car.setId(9943);
car.setName("Honda");
Person person = new Person();
person.setId(123);
person.setName("Jon");
person.setCar(car);
Model model = new Model();
model.setCar(car);
model.setPerson(person);
return model;
}
private static Gson createGson() {
Set<Class> classes = new HashSet<>();
classes.add(Car.class);
classes.add(Person.class);
return new GsonBuilder()
.setPrettyPrinting()
.registerTypeAdapterFactory(new CachedInstancesTypeAdapterFactory(classes))
.create();
}
}
上面的代码打印,首先是JSON
:
{
"car": {
"id": 9943,
"name": "Honda"
},
"person": {
"id": 123,
"name": "Jon",
"car": {
"id": 9943,
"name": "Honda"
}
}
}
然后:
Model{car=Car{id=9943, name='Honda'}, person=Person{id=123, name='Jon', car=Car{id=9943, name='Honda'}}}
Two car instances are the same: true
CachedInstancesTypeAdapterFactory
实现不是线程安全的。此外,当您想要使用 Car
和 反序列化
实例。原因是 JSON
负载时,您必须始终为每个线程和每次尝试创建新的 Gson
对象PersonCachedInstancesTypeAdapterFactory#cachedMaps
对象只能使用一次。
另见:
关于java - GSON:在两个类中引用相同的对象,解码后实例重复,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58841054/
我试图让 gson 在 java 项目上工作,但每次运行时都会出现上述错误。我没有使用任何闪存 IDE,只是 vim,我看到的与我的问题相关的每个问题都与 eclipse 及其部署设置有关。我希望有人
我一直在关注将对象数组转换为 JSON 的教程,但我遇到了一些找不到解决方案的错误。 代码 第 60 - 64 行 Gson gson = new Gson().toJson(data); respo
我正在创建一个新的 Netty 管道,我正在尝试: 避免过早优化。 编写易于向我的一位实习生解释的代码。 这种工厂方法当然很容易解释: public String toJSON() { Gso
我尝试在程序中使用 Unirest,但不断收到此错误 java.lang.NoSuchMethodError: com.google.gson.Gson.newBuilder()Lcom/google
// Retrofit compile 'com.squareup.retrofit2:retrofit:2.1.0' // JSON Parsing compile 'com.google.code
我正在尝试将 Gson 与接口(interface)一起使用: public interface Photo { public int getWidth(); } public class D
我正在执行一个 Web 服务,这是我的响应,但是当我尝试使用 Gson 库将此 JSONObject(org.json.JSONObject) 转换为特定对象时,我的应用程序崩溃了。所以,我不知道为什
我有以下 .json 文件 - { "IDs":[ "1716136233", "2030187302", "204897807
我试图在我的项目中使用 GSON,但我的应用程序崩溃了,logcat 说找不到 com.google.gson.Gson。我已将 import com.google.gson.Gson 放在我的类文件
在下面的示例中,我尝试序列化我的自定义类Couple,其中包含Point2D 类型的字段。 我发现,SErialzing 的过程取决于 DEserializer 的存在。 此外,如果存在反序列化器,则
我有课 class ThreadComment( banned: Int, closed: Int, comment: String?, date: String?,
使用 gson 解析 json 时,我有一个奇怪的行为。我使用这个代码: private static Container parseContainer(String containerJson) {
我注意到 GSON HTML 转义 字符,可以通过使用disableHtmlEscaping()来禁用它构建器配置方法。但我的问题是 - 为什么 GSON 默认情况下会进行 HTML 转义?不进行 H
我注意到一个奇怪的问题。我可以使用 Junit 运行我的测试用例,但是当我使用 maven 运行时,其中一个测试用例失败。它提示找不到 Gson 类 def。 我能够在 Maven 依赖项中看到 Gs
我有一个看起来像这样的 json {response:{"status":{"....."},data:[{"name":"Alice","id":"123"},{"name":"Jack","id"
所以,我的应用已经发布了将近一年,但没有看到这个问题,现在它出现了。 即使是现在,我的手机上的调试版本也没有这个问题。我对从 Android Studio 打开的任何模拟器没有任何问题。然而,Goog
我是 Android 开发环境的新手,因此需要专家的帮助。 java.lang.NoClassDefFoundError: com.google.gson.Gson 项目中包含 Gson 库,但是当从
总结 我在我的 android 应用程序中使用 Retrofit 和 Gson 有一段时间了,但是我从服务器获得的 ID 是 13 位数字。 Gson 自动将这些数字转换为科学计数法,然后将其转换为
我正在使用一个大的 JSON 对象,它具有来自多个请求的响应。 我正在处理的部分只需要很少的对象,而且它们并不总是在前面。例如 json 结构是: ** json = { mainDocume
我有一个休息网络服务: @Path("/tranreq") public class TranscriptRequesterResource { private static final Gs
我是一名优秀的程序员,十分优秀!