- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在构建 Android 应用,它使用 JSON 与 C# 服务器通信。
我有一个要序列化的数据类(实际上是将接收到的数据反序列化到其中),其中包含派生类的 List 字段,但将它们存储为它们的基类:
public class ToSerializeClass{
@SerializedName("TestString")
private String testString = "TestStringValue";
@SerializedName("DerivedClasses")
private List<BaseClass> derivedClasses;
public List<BaseClass> getDerivedClasses() {
return derivedClasses;
}
public ToSerializeClass(List<BaseClass> derivedClasses){
this.derivedClasses= derivedClasses;
}
}
例如,从 C# 端,如果我收到以下 ToSerializeClass 实例,名为 serializeClass:
List<BaseClass> derivedClasses = new ArrayList<>();
derivedClasses.add(new DerivedClassA());
derivedClasses.add(new DerivedClassB());
ToSerializeClass serializeClass = new ToSerializeClass(derivedClasses);
JSON 字符串将是:
{"__type":"ToSerializeClass","DerivedClasses":[{"__type":"DerivedA","FieldA":"This is a derived class."},{"__type":"DerivedB","FieldB":"This is ANOTHER derived class.", "IntValue":10}],"TestString":"TestStringValue"}
JSON 字符串的 "__type":"SimpleClassName" 字段显示序列化类的简单名称。这些是由 C# 端的序列化器添加的。如有必要,我可以使这些字段消失,但这是 C# 端同一问题的解决方案。如果没有类型,它看起来像这样:
{"DerivedClasses":[{"FieldA":"This is a derived class."},{"FieldB":"This is ANOTHER derived class.", "IntValue":10}],"TestString":"TestStringValue"}
问题是,当我尝试将 JSON 字符串反序列化为 ToSerializeClass 类的实例时:
Gson serializer = new Gson();
ToSerializeClass deserialized = serializer.fromJson(jsonString, ToSerializeClass.class);
我有一个反序列化类,它是一个ToSerializeClass实例,但是衍生类列表是基类而不是派生类的集合,并且所有派生信息都会丢失。
如何将字符串反序列化为 ToSerializeClass 实例以获得派生类列表?
我可以完全控制源代码,因此我可以修改我的数据类,必要时使用不同的集合,创建一些包装器,修改 JSON 字符串,但我想解决它如果可能的话使用 Gson,并且不需要太多开销即可完成。
谢谢!
编辑:例如DerivedClassA和DerivedClassB:
public class DerivedClassA extends BaseClass{
@SerializedName("FieldA")
private String fieldA = "This is a derived class.";
...
public DerivedClassA(){
super();
}
}
public class DerivedClassB extends BaseClass{
@SerializedName("FieldB")
private String fieldB = "This is ANOTHER derived class.";
@SerializedName("IntValue")
private int intValue = 10;
...
public DerivedClassB(){
super();
}
}
最佳答案
我创建了一个快速解决方案,它接近我正在寻找的解决方案(我不是 Java 人员,所以如果它伤害了您的大脑,我很抱歉,请随时发表评论建议)。
数据类:
public class BaseClass {
@SerializedName("Method")
private String method;
public void setMethod(String method){
this.method = method;
}
public String getMethod(){
return method;
}
public BaseClass(String method){
this.method = method;
}
}
public class DerivedClassA extends BaseClass{
@SerializedName("FieldA")
private String fieldA = "This is a derived class.";
public DerivedClassA(){
super("ClassA");
}
}
public class DerivedClassB extends BaseClass{
@SerializedName("FieldB")
private String fieldB = "This is ANOTHER derived class.";
@SerializedName("IntValue")
private int intValue = 10;
public DerivedClassB(){
super("ClassB");
}
}
public class ToSerializeClass{
@SerializedName("TestString")
private String testString = "TestStringValue";
@SerializedName("DerivedClasses")
private List<BaseClass> derivedClasses;
public ToSerializeClass(List<BaseClass> derivedClasses){
this.derivedClasses= derivedClasses;
}
}
解决方案:ClassDeserializerAdapter实现:
public class ClassDeserializerAdapter implements JsonDeserializer<BaseClass>
{
private String typeName;
private Gson gson;
private Map<String, Class<? extends BaseClass>> classTypeRegistry;
ClassDeserializerAdapter(String typeName)
{
this.typeName = typeName;
gson = new Gson();
classTypeRegistry = new HashMap<>();
}
void registerClassType(String classTypeName, Class<? extends BaseClass> classType)
{
// registering Types to Strings
classTypeRegistry.put(classTypeName, classType);
}
@Override
public BaseClass deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context)
throws JsonParseException
{
JsonObject jsonObject = json.getAsJsonObject();
JsonElement typeElement = jsonObject.get(typeName);
String method = typeElement.getAsString();
Class<? extends BaseClass> classType = classTypeRegistry.get(method);
BaseClass result = gson.fromJson(json, classType);
return result;
}
}
这对我来说效果很好,甚至不需要临时类(class)等。
注意:通过反射,它可以自动注册从 BaseClass 继承的每个类,您甚至不需要处理注册过程。
示例:main()方法的主体是:
// **SERIALIZATION PART** (nothing special, simple Gson serialization)
// Creating a list to pass as parameter to the container class
List<BaseClass> derivedClasses = new ArrayList<>();
derivedClasses.add(new DerivedClassA());
derivedClasses.add(new DerivedClassB());
// Creating the container class to be serialized
ToSerializeClass serializeClass = new ToSerializeClass(derivedClasses);
Gson gson = new Gson();
String json = gson.toJson(serializeClass);
// json = {"TestString":"TestStringValue","DerivedClasses":[{"FieldA":"This is a derived class.","Method":"ClassA"},{"FieldB":"This is ANOTHER derived class.","IntValue":10,"Method":"ClassB"}]}
// **DESERIALIZATION PART** (with custom deserializer)
// creating the custom deserializer, which will find the derived class' type as the class' "Method" field value. With that value, it can resolve the type.. see below
ClassDeserializerAdapter deserializer = new ClassDeserializerAdapter("Method");
// registering each Type into the Deserializer's HashMap (key-value pair), where the key (String) must be carried by the object (you can find it in the BaseClass, called "Method")
deserializer.registerClassType("ClassA", DerivedClassA.class);
deserializer.registerClassType("ClassB", DerivedClassB.class);
Gson gsonB = new GsonBuilder().registerTypeAdapter(BaseClass.class, deserializer).create();
// deserializing
ToSerializeClass deserialized = gsonB.fromJson(json, ToSerializeClass.class); // CORRECT!
关于java - 使用 Gson 将基类列表反序列化为派生类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33857091/
我试图让 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
我是一名优秀的程序员,十分优秀!