![](/img/trans.png)
[英]adding an object using a custom typeadapter, jsonwriter in gson
[英]Gson custom seralizer for one variable (of many) in an object using TypeAdapter
我见过很多使用自定义 TypeAdapter 的简单示例。 最有用的是Class TypeAdapter<T>
。 但这还没有回答我的问题。
我想自定义对象中单个字段的序列化,让默认的 Gson 机制来处理剩下的事情。
出于讨论的目的,我们可以使用这个类定义作为我希望序列化的对象的类。 我想让 Gson 序列化前两个类成员以及基类的所有公开成员,并且我想对下面显示的第三个也是最后一个类成员进行自定义序列化。
public class MyClass extends SomeClass {
@Expose private HashMap<String, MyObject1> lists;
@Expose private HashMap<String, MyObject2> sources;
private LinkedHashMap<String, SomeClass> customSerializeThis;
[snip]
}
这是一个很好的问题,因为它隔离了一些应该很容易但实际上需要大量代码的东西。
首先,编写一个抽象的TypeAdapterFactory
,它为您提供修改传出数据的钩子。 此示例使用 Gson 2.2 中名为getDelegateAdapter()
的新 API,它允许您查找 Gson 默认使用的适配器。 如果您只想调整标准行为,委托适配器非常方便。 与完全自定义类型的适配器不同,它们会在您添加和删除字段时自动保持最新状态。
public abstract class CustomizedTypeAdapterFactory<C>
implements TypeAdapterFactory {
private final Class<C> customizedClass;
public CustomizedTypeAdapterFactory(Class<C> customizedClass) {
this.customizedClass = customizedClass;
}
@SuppressWarnings("unchecked") // we use a runtime check to guarantee that 'C' and 'T' are equal
public final <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
return type.getRawType() == customizedClass
? (TypeAdapter<T>) customizeMyClassAdapter(gson, (TypeToken<C>) type)
: null;
}
private TypeAdapter<C> customizeMyClassAdapter(Gson gson, TypeToken<C> type) {
final TypeAdapter<C> delegate = gson.getDelegateAdapter(this, type);
final TypeAdapter<JsonElement> elementAdapter = gson.getAdapter(JsonElement.class);
return new TypeAdapter<C>() {
@Override public void write(JsonWriter out, C value) throws IOException {
JsonElement tree = delegate.toJsonTree(value);
beforeWrite(value, tree);
elementAdapter.write(out, tree);
}
@Override public C read(JsonReader in) throws IOException {
JsonElement tree = elementAdapter.read(in);
afterRead(tree);
return delegate.fromJsonTree(tree);
}
};
}
/**
* Override this to muck with {@code toSerialize} before it is written to
* the outgoing JSON stream.
*/
protected void beforeWrite(C source, JsonElement toSerialize) {
}
/**
* Override this to muck with {@code deserialized} before it parsed into
* the application type.
*/
protected void afterRead(JsonElement deserialized) {
}
}
上面的类使用默认序列化来获取一个 JSON 树(由JsonElement
表示),然后调用钩子方法beforeWrite()
以允许子类自定义该树。 与afterRead()
反序列化类似。
接下来,我们为特定的MyClass
示例将其子类化。 为了说明,我将在地图序列化时向地图添加一个名为“大小”的综合属性。 为了对称,我会在反序列化时将其删除。 在实践中,这可以是任何定制。
private class MyClassTypeAdapterFactory extends CustomizedTypeAdapterFactory<MyClass> {
private MyClassTypeAdapterFactory() {
super(MyClass.class);
}
@Override protected void beforeWrite(MyClass source, JsonElement toSerialize) {
JsonObject custom = toSerialize.getAsJsonObject().get("custom").getAsJsonObject();
custom.add("size", new JsonPrimitive(custom.entrySet().size()));
}
@Override protected void afterRead(JsonElement deserialized) {
JsonObject custom = deserialized.getAsJsonObject().get("custom").getAsJsonObject();
custom.remove("size");
}
}
最后通过创建一个使用新类型适配器的自定义Gson
实例将它们放在一起:
Gson gson = new GsonBuilder()
.registerTypeAdapterFactory(new MyClassTypeAdapterFactory())
.create();
Gson 新的TypeAdapter和TypeAdapterFactory类型非常强大,但它们也是抽象的,需要实践才能有效地使用。 希望你觉得这个例子很有用!
还有另一种方法。 正如 Jesse Wilson 所说,这应该很容易。 猜猜看,这很容易!
如果您为您的类型实现JsonSerializer
和JsonDeserializer
,您可以处理您想要的部分并将其他所有内容委托给 Gson ,只需很少的代码。 为方便起见,我在下面的另一个问题中引用了@Perception 的回答,有关详细信息,请参阅该答案:
在这种情况下,最好使用
JsonSerializer
而不是TypeAdapter
,原因很简单,序列化程序可以访问其序列化上下文。public class PairSerializer implements JsonSerializer<Pair> { @Override public JsonElement serialize(final Pair value, final Type type, final JsonSerializationContext context) { final JsonObject jsonObj = new JsonObject(); jsonObj.add("first", context.serialize(value.getFirst())); jsonObj.add("second", context.serialize(value.getSecond())); return jsonObj; } }
这样做的主要优点(除了避免复杂的解决方法)是您仍然可以利用可能已在主上下文中注册的其他类型适配器和自定义序列化程序。 请注意,序列化程序和适配器的注册使用完全相同的代码。
但是,我承认如果您经常修改 Java 对象中的字段,Jesse 的方法看起来会更好。 这是易用性与灵活性之间的权衡,任您选择。
我同事也提到了@JsonAdapter
注解的使用
https://google.github.io/gson/apidocs/com/google/gson/annotations/JsonAdapter.html
该页面已移至此处: https : //www.javadoc.io/doc/com.google.code.gson/gson/latest/com.google.gson/com/google/gson/annotations/JsonAdapter.html
例子:
private static final class Gadget {
@JsonAdapter(UserJsonAdapter2.class)
final User user;
Gadget(User user) {
this.user = user;
}
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.