我目前遇到了某些内部对象反序列化的问题,在
Spring中,我在使用@ResponseBody输出它们之前初始化所有对象.
举个例子,这是一个回应:
[{id:1, location:{id:1, ... extra location data}},
{id:2, location:1}
]
现在,GSON抛出一个错误,因为它无法理解该位置:1指的是在前一个对象中已经反序列化的位置对象.
反序列化按以下方法完成:
@Override
public void handleReader(Reader reader) {
try {
String json = readerToString(reader);
T object = getGson().fromJson(json, returnType);
handleObject(object);
} catch (Exception e) {
Sentry.captureException(e);
}
}
例如,这是通过常规泛型类调用的,我使用类型Event []作为T泛型以返回数组.
我怎样才能使用Gson修复此问题或每次弹出输出完整数据?理想情况下,我想修复Gson,因为这样可以大幅降低带宽,但此时我并不太感兴趣.
我的Spring返回方法如下:
@Override
public List<T> list() {
return service.findAll();
}
初始化如下:
@Override
@Transactional
public List<Event> findAll() {
List<Event> list = eventRepository.findByArchivedFalse();
for (Event event : list) {
this.initialize(event);
}
return list;
}
@Override
public Event initialize(Event obj) {
Hibernate.initialize(obj.getLocation());
Hibernate.initialize(obj.getLocation().get... inner data here);
return obj;
}
我想这需要一个真正的结构审查,但是,如果我能帮助它,我想保持结构大致相同.
最佳答案 如果你不愿意改变JSon,你将不得不编写自定义反序列化器.但是,更改JSon正是我建议的.
选项1:更改JSon
我认为正确的做法是有两个单独的消息,例如
{
"uniqueLocations":
[
{"id":1, ... extra location details} ,
],
"locationMap":
[
{"id":1,"location":1},
{"id":2,"location":1}
... etc.
]
}
这更清楚;这会将您的json分开,以便您在相同位置始终拥有相同类型的数据.
选项2:让Gson能够进行更复杂的反序列化
但是,如果您不愿意这样做,您可以编写自定义反序列化程序.最直接的方法是扩展TypeAdapter
,只使用特定的具体类,而不是参数化类型.但是,如果要使用参数化类型,则必须使用TypeAdapterFactory
.
您可以在此处详细了解如何执行此操作:How do I implement TypeAdapterFactory in Gson?