AFAIK可以通过TypeAdapterFactory
进行最灵活的gson自定义,但它可能会变得不必要地复杂化。它强迫我为每个被处理的类read
和write
编写,而有时只需要一种方法。此外,有时JsonSerializer
和/或JsonDeserializer
更容易编写,例如like here。这引出了我这些问题:
TypeAdapter
来简单地委托其中一种方法(例如将ImmutableList
写为List
)?JsonSerializer
和/或JsonDeserializer
与TypeAdapterFactory
一起使用?或者,他们有工厂吗?答案 0 :(得分:16)
可以创建一个委托其中一个方法的TypeAdapter
。这个用例是API的一个重要部分,并且有一个getDelegateAdapter()方法就是为了这个目的。将this
作为getDelegateAdapter
的第一个参数传递,它将返回当前工厂之后优先的适配器。
TypeAdapterFactory immutableListFactory = new TypeAdapterFactory() {
@Override public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
if (!(type.getType() instanceof ParameterizedType)
|| !type.getRawType().equals(ImmutableList.class)) {
return null;
}
ParameterizedType parameterizedType = (ParameterizedType) type.getType();
TypeAdapter<T> delegate = gson.getDelegateAdapter(this, type);
TypeAdapter<?> elementAdapter = gson.getAdapter(
TypeToken.get(parameterizedType.getActualTypeArguments()[0]));
return new ImmutableListAdapter(delegate, elementAdapter);
}
class ImmutableListAdapter<E> extends TypeAdapter<ImmutableList<E>> {
private TypeAdapter<List<E>> delegate;
private TypeAdapter<E> element;
ImmutableListAdapter(TypeAdapter<List<E>> delegate, TypeAdapter<E> element) {
this.delegate = delegate;
this.element = element;
}
@Override public void write(JsonWriter out, ImmutableList<E> value) throws IOException {
delegate.write(out, value);
}
@Override public ImmutableList<E> read(JsonReader in) throws IOException {
if (in.peek() == JsonToken.NULL) {
in.nextNull();
return null;
}
ImmutableList.Builder<E> builder = ImmutableList.builder();
in.beginArray();
while (in.hasNext()) {
builder.add(element.read(in));
}
in.endArray();
return builder.build();
}
}
};
您可以将JsonSerializer
/ JsonDeserializer
与TypeAdapterFactory
混合搭配,但不能直接匹配。最简单的方法是回调Gson以序列化类中的子值。在这个例子中,我们将内部循环更改为:
while (in.hasNext()) {
builder.add(gson.<E>fromJson(in, elementType));
}
JsonSerializer
/ JsonDeserializer
和TypeAdapter
之间的主要区别在于从JSON到对象模型需要多少个阶段。使用JsonSerializer
/ JsonDeserializer
对象首先转换为Gson的DOM模型(JsonElement
等),然后转换为对象模型。使用TypeAdapter
,将跳过中间步骤。
这使得类型适配器代码的读写变得有点棘手,所以你应该只选择它来优化代码。