IllegalStateException - 使用 Gson 序列化带有字符串数组的地图
IllegalStateException - Serializing a map with a String array using 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
}
我还有其他以类似方式编码和解码的地图,但这是唯一给我带来问题的地图。有没有办法让这个工作?
也许你应该为String[]
写一个toString()
方法,否则你只会得到每个String[]
的地址
很遗憾,您无法更改数据结构中元素的类型。您应该永远不要将数组存储为键,因为它们不会覆盖 equals、hashcode 和 toString。
话虽如此,让我们谈谈您的错误。这是因为Gson将key序列化为String,因为JSON语法要求在键值对中,key必须是字符串。但是您期望数组作为键,因此解析器会抛出错误。
由于您无法更改结构,您可以编写自定义序列化程序和反序列化程序来处理此问题:
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
返回 "same" 映射(我引用 "same" 因为根据 Arrays.equals
而不是 .equals()
,键是相等的)。
希望对您有所帮助!
这是我收到的完整错误:
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
}
我还有其他以类似方式编码和解码的地图,但这是唯一给我带来问题的地图。有没有办法让这个工作?
也许你应该为String[]
写一个toString()
方法,否则你只会得到每个String[]
很遗憾,您无法更改数据结构中元素的类型。您应该永远不要将数组存储为键,因为它们不会覆盖 equals、hashcode 和 toString。
话虽如此,让我们谈谈您的错误。这是因为Gson将key序列化为String,因为JSON语法要求在键值对中,key必须是字符串。但是您期望数组作为键,因此解析器会抛出错误。
由于您无法更改结构,您可以编写自定义序列化程序和反序列化程序来处理此问题:
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
返回 "same" 映射(我引用 "same" 因为根据 Arrays.equals
而不是 .equals()
,键是相等的)。
希望对您有所帮助!