IllegalStateException - 使用Gson使用String数组序列化映射

时间:2015-04-05 23:27:19

标签: java hashmap gson

以下是我收到的完整错误:

com.google.gson.JsonSyntaxException: java.lang.IllegalStateException: Expected BEGIN_ARRAY but was STRING at line 1 column 3 path $.

我想要编码的地图是

HashMap<String[],Boolean>.

我用它来序列化地图:

Gson gson = new Gson();
gson.toJson(object);

这将反序列化对象:

json.fromJson(encodedObject, new TypeToken<HashMap<String[], Boolean>>(){}.getType())

这是原始JSON输出:

{
     "[Ljava.lang.String;@433fe238":false,
     "[Ljava.lang.String;@433feb38":false,
     "[Ljava.lang.String;@433fe018":false,
     "[Ljava.lang.String;@433fd618":false
}

我有其他地图,我以类似的方式编码和解码,但这是唯一一个给我带来问题的地图。有没有办法使这项工作?

2 个答案:

答案 0 :(得分:2)

很遗憾,您无法更改数据结构中的元素类型。您可能永远不会将数组存储为键,因为它们不会覆盖equals,hashcode和toString。

那说让我们谈谈你的错误。这是因为Gson将密钥序列化为String,因为JSON语法要求在键值对中,密钥必须是字符串。但是你期望一个数组作为键,因此解析器会抛出错误。

由于您无法更改结构,因此您可以编写自定义序列化程序和反序列化程序来处理此问题:

class MapSerializer implements JsonSerializer<Map<String[], Boolean>> {
    @Override
    public JsonElement serialize(Map<String[], Boolean> src, Type typeOfSrc, JsonSerializationContext context) {
        JsonObject obj = new JsonObject();
        for(Map.Entry<String[], Boolean> entry : src.entrySet()) {
            obj.addProperty(Arrays.toString(entry.getKey()), entry.getValue());
        }
        return obj;
    }
}

class MapDeserializer implements JsonDeserializer<Map<String[], Boolean>> {
    @Override
    public Map<String[], Boolean> deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
        Map<String[],Boolean> map = new HashMap<>();
        JsonObject obj = json.getAsJsonObject();
        for(Map.Entry<String, JsonElement> entry : obj.entrySet()) {
            String s = entry.getKey();
            //See how you're dependent of the String representation given by Arrays.toString? 
            //Not very convenient.
            map.put(s.substring(1, s.length()-1).split(", "), entry.getValue().getAsBoolean());
        }
        return map;
    }
}

基本上,密钥现在由Arrays.toString给出,toString是数组中元素的可读表示,而不是Arrays.toString方法。

但正如您所看到的,反序列化过程必须假设此表示不会发生变化,这是非常危险的,因为Map<String[],Boolean> map = new HashMap<>(); map.put(new String[]{"Hello"},false); map.put(new String[]{"Stack", "Overflow"},true); Type t = new TypeToken<Map<String[], Boolean>>(){}.getType(); Gson gson = new GsonBuilder().registerTypeAdapter(t, new MapSerializer()) .registerTypeAdapter(t, new MapDeserializer()) .setPrettyPrinting() .create(); String repr = gson.toJson(map, t); Map<String[], Boolean> map2 = gson.fromJson(repr, t); 的结果中的任何修改都会破坏该过程,但您必须处理它......(虽然它不太可能发生,但这是你必须要注意的事实)

要使其工作,您需要在解析器中注册适配器。

repr

如果您尝试打印{ "[Hello]": false, "[Stack, Overflow]": true } ,您将获得:

fromJson

比原始输出更好。你回来了#34;同样的#34;使用Arrays.equals制作地图(我引用&#34;相同&#34;因为根据.equals()而不是{{1}},键是等于的。)

希望它有所帮助!

答案 1 :(得分:0)

也许你应该为toString()编写一个String[]方法,否则你只能得到每个String的地址[]