我想通过谷歌Gson传输一个列表对象,但我不知道如何反序列化泛型类型。

看完这篇文章(BalusC的回答)后,我尝试的是:

MyClass mc = new Gson().fromJson(result, new List<MyClass>() {}.getClass());

但是然后我在Eclipse中得到一个错误,说“类型新列表<MyClass>(){}必须实现继承的抽象方法…”,如果我使用一个快速修复,我得到了一个超过20个方法存根的怪物。

我很确定有一个更简单的解决办法,但我似乎找不到!

现在我有了这个:

Type listType = new TypeToken<List<MyClass>>() {}.getType();

MyClass mc = new Gson().fromJson(result, listType);

然而,我确实在fromJson行得到以下异常:

java.lang.NullPointerException
at org.apache.harmony.luni.lang.reflect.ListOfTypes.length(ListOfTypes.java:47)
at org.apache.harmony.luni.lang.reflect.ImplForType.toString(ImplForType.java:83)
at java.lang.StringBuilder.append(StringBuilder.java:203)
at com.google.gson.JsonDeserializerExceptionWrapper.deserialize(JsonDeserializerExceptionWrapper.java:56)
at com.google.gson.JsonDeserializationVisitor.invokeCustomDeserializer(JsonDeserializationVisitor.java:88)
at com.google.gson.JsonDeserializationVisitor.visitUsingCustomHandler(JsonDeserializationVisitor.java:76)
at com.google.gson.ObjectNavigator.accept(ObjectNavigator.java:106)
at com.google.gson.JsonDeserializationContextDefault.fromJsonArray(JsonDeserializationContextDefault.java:64)
at com.google.gson.JsonDeserializationContextDefault.deserialize(JsonDeserializationContextDefault.java:49)
at com.google.gson.Gson.fromJson(Gson.java:568)
at com.google.gson.Gson.fromJson(Gson.java:515)
at com.google.gson.Gson.fromJson(Gson.java:484)
at com.google.gson.Gson.fromJson(Gson.java:434)

我做捕获JsonParseExceptions和结果不是空。

我用调试器检查了listType,得到了以下结果:

列表类型 args = ListOfTypes List = null resolvedTypes =类型[1] loader = PathClassLoader ownerType0 = null ownerTypeRes = null rawType =类(java.util.ArrayList) rawTypeName = "java.util.ArrayList"

因此,getClass调用似乎没有正常工作。有什么建议吗?

我查过Gson用户指南。它提到了在将泛型类型解析为Json时应该发生的运行时异常。我做得“错误”(上面没有显示),就像在示例中一样,但根本没有得到异常。所以我按照用户指南的建议改变了序列化。不过,这也无济于事。

编辑:

解决了,请看下面我的答案。


当前回答

我想再补充一种可能性。如果你不想使用TypeToken,并想将json对象数组转换为数组列表,那么你可以这样做:

如果你的json结构像这样:

{

"results": [
    {
        "a": 100,
        "b": "value1",
        "c": true
    },
    {
        "a": 200,
        "b": "value2",
        "c": false
    },
    {
        "a": 300,
        "b": "value3",
        "c": true
    }
]

}

你的类结构是这样的:

public class ClassName implements Parcelable {

    public ArrayList<InnerClassName> results = new ArrayList<InnerClassName>();
    public static class InnerClassName {
        int a;
        String b;
        boolean c;      
    }
}

然后你可以这样解析它:

Gson gson = new Gson();
final ClassName className = gson.fromJson(data, ClassName.class);
int currentTotal = className.results.size();

现在您可以访问className对象的每个元素了。

其他回答

另一种方法是使用数组作为类型,例如:

MyClass[] mcArray = gson.fromJson(jsonString, MyClass[].class);

这样你就避免了Type对象的所有麻烦,如果你真的需要一个列表,你可以通过以下方式将数组转换为列表:

List<MyClass> mcList = Arrays.asList(mcArray);

恕我直言,这样可读性更强。

为了使它成为一个实际的列表(可以修改,请参阅Arrays.asList()的限制),然后只需执行以下操作:

List<MyClass> mcList = new ArrayList<>(Arrays.asList(mcArray));

我想再补充一种可能性。如果你不想使用TypeToken,并想将json对象数组转换为数组列表,那么你可以这样做:

如果你的json结构像这样:

{

"results": [
    {
        "a": 100,
        "b": "value1",
        "c": true
    },
    {
        "a": 200,
        "b": "value2",
        "c": false
    },
    {
        "a": 300,
        "b": "value3",
        "c": true
    }
]

}

你的类结构是这样的:

public class ClassName implements Parcelable {

    public ArrayList<InnerClassName> results = new ArrayList<InnerClassName>();
    public static class InnerClassName {
        int a;
        String b;
        boolean c;      
    }
}

然后你可以这样解析它:

Gson gson = new Gson();
final ClassName className = gson.fromJson(data, ClassName.class);
int currentTotal = className.results.size();

现在您可以访问className对象的每个元素了。

关于Gson的“Type”类理解,请参考示例2。

例1:在这个deserilizeResturant中,我们使用Employee[]数组并获取详细信息

public static void deserializeResturant(){

       String empList ="[{\"name\":\"Ram\",\"empId\":1},{\"name\":\"Surya\",\"empId\":2},{\"name\":\"Prasants\",\"empId\":3}]";
       Gson gson = new Gson();
       Employee[] emp = gson.fromJson(empList, Employee[].class);
       int numberOfElementInJson = emp.length();
       System.out.println("Total JSON Elements" + numberOfElementInJson);
       for(Employee e: emp){
           System.out.println(e.getName());
           System.out.println(e.getEmpId());
       }
   }

示例2:

//Above deserilizeResturant used Employee[] array but what if we need to use List<Employee>
public static void deserializeResturantUsingList(){

    String empList ="[{\"name\":\"Ram\",\"empId\":1},{\"name\":\"Surya\",\"empId\":2},{\"name\":\"Prasants\",\"empId\":3}]";
    Gson gson = new Gson();

    // Additionally we need to se the Type then only it accepts List<Employee> which we sent here empTypeList
    Type empTypeList = new TypeToken<ArrayList<Employee>>(){}.getType();


    List<Employee> emp = gson.fromJson(empList, empTypeList);
    int numberOfElementInJson = emp.size();
    System.out.println("Total JSON Elements" + numberOfElementInJson);
    for(Employee e: emp){
        System.out.println(e.getName());
        System.out.println(e.getEmpId());
    }
}

因为它回答了我最初的问题,我已经接受了doc_180的答案,但如果有人再次遇到这个问题,我也会回答我问题的后半部分:

我所描述的NullPointerError与列表本身无关,而是与它的内容有关!

“MyClass”类没有“no args”构造函数,它的超类也没有。一旦我向MyClass及其超类添加了一个简单的“MyClass()”构造函数,一切都工作得很好,包括doc_180建议的List序列化和反序列化。

下面是一个使用动态定义类型的解决方案。诀窍在于使用array . newinstance()创建合适的数组类型。

public static <T> List<T> fromJsonList(String json, Class<T> clazz) {
    Object [] array = (Object[])java.lang.reflect.Array.newInstance(clazz, 0);
    array = gson.fromJson(json, array.getClass());
    List<T> list = new ArrayList<T>();
    for (int i=0 ; i<array.length ; i++)
        list.add(clazz.cast(array[i]));
    return list; 
}