Это прискорбно вы не можете изменить тип элементов структуры данных. Вероятно, вы никогда не должны хранить массивы в качестве ключей, поскольку они не переопределяют равные, 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
(я процитировал «тот же», потому что ключи равны Arrays.equals
не .equals()
).
Надеюсь, это поможет!
Я не могу помочь вам с проблемой Gson, поскольку у меня нет опыта в этом, но я вижу одну важную проблему, которую вы, возможно, захотите решить. Массив нельзя использовать очень хорошо, как ключ 'HashMap'. Массивы не имеют корректного поведения 'hashCode()' и 'equals()'. Используйте «Список'. –
В любом случае, я Googled, и ваша проблема кажется такой же: http://stackoverflow.com/questions/25522309/converting-json-between-string-and-byte-with-gson –
Спасибо за головы. Этот старый код, который я пытаюсь исправить, и я не могу изменить структуру данных без каких-либо проблем. Я был немного новичком, когда я его написал :) – eBehbahani