This Gsonプロジェクトからのリンクは、型付きマップをJSONにシリアル化するために次のようなことをする必要があることを示しているようです。
public static class NumberTypeAdapter
implements JsonSerializer<Number>, JsonDeserializer<Number>,
InstanceCreator<Number> {
public JsonElement serialize(Number src, Type typeOfSrc, JsonSerializationContext
context) {
return new JsonPrimitive(src);
}
public Number deserialize(JsonElement json, Type typeOfT,
JsonDeserializationContext context)
throws JsonParseException {
JsonPrimitive jsonPrimitive = json.getAsJsonPrimitive();
if (jsonPrimitive.isNumber()) {
return jsonPrimitive.getAsNumber();
} else {
throw new IllegalStateException("Expected a number field, but was " + json);
}
}
public Number createInstance(Type type) {
return 1L;
}
}
public static void main(String[] args) {
Map<String, Number> map = new HashMap<String, Number>();
map.put("int", 123);
map.put("long", 1234567890123456789L);
map.put("double", 1234.5678D);
map.put("float", 1.2345F);
Type mapType = new TypeToken<Map<String, Number>>() {}.getType();
Gson gson = new GsonBuilder().registerTypeAdapter(Number.class, new
NumberTypeAdapter()).create();
String json = gson.toJson(map, mapType);
System.out.println(json);
Map<String, Number> deserializedMap = gson.fromJson(json, mapType);
System.out.println(deserializedMap);
}
クールで動作しますが、オーバーヘッドが大きいようです(Type Adapterクラス全体?) 。 JSONLibのような他のJSONライブラリを使用しており、次の方法でマップを作成できます。
JSONObject json = new JSONObject();
for(Entry<String,Integer> entry : map.entrySet()){
json.put(entry.getKey(), entry.getValue());
}
または、次のようなカスタムクラスがある場合:
JSONObject json = new JSONObject();
for(Entry<String,MyClass> entry : map.entrySet()){
JSONObject myClassJson = JSONObject.fromObject(entry.getValue());
json.put(entry.getKey(), myClassJson);
}
このプロセスはより手作業ですが、必要なコードが少なく、Number用のカスタムタイプアダプターを作成するためのオーバーヘッドや、ほとんどの場合for my独自のカスタムクラス。
これが、Gsonでマップをシリアル化する唯一の方法ですか、または上記のリンクでGsonが推奨する方法を打ち破る方法を誰かが見つけましたか?.
必要なのはTypeToken
部分のみです(ジェネリックが含まれる場合)。
Map<String, String> myMap = new HashMap<String, String>();
myMap.put("one", "hello");
myMap.put("two", "world");
Gson gson = new GsonBuilder().create();
String json = gson.toJson(myMap);
System.out.println(json);
Type typeOfHashMap = new TypeToken<Map<String, String>>() { }.getType();
Map<String, String> newMap = gson.fromJson(json, typeOfHashMap); // This type must match TypeToken
System.out.println(newMap.get("one"));
System.out.println(newMap.get("two"));
出力:
{"two":"world","one":"hello"}
hello
world
Mapシリアル化のデフォルトのGson実装では、キーでtoString()
を使用します。
Gson gson = new GsonBuilder()
.setPrettyPrinting().create();
Map<Point, String> original = new HashMap<>();
original.put(new Point(1, 2), "a");
original.put(new Point(3, 4), "b");
System.out.println(gson.toJson(original));
あげる:
{
"Java.awt.Point[x\u003d1,y\u003d2]": "a",
"Java.awt.Point[x\u003d3,y\u003d4]": "b"
}
enableComplexMapKeySerialization
を使用するデフォルトのGsonルールに従ってマップキーをシリアル化する場合は、 enableComplexMapKeySerialization を使用できます。これは、キーと値のペアの配列の配列を返します。
Gson gson = new GsonBuilder().enableComplexMapKeySerialization()
.setPrettyPrinting().create();
Map<Point, String> original = new HashMap<>();
original.put(new Point(1, 2), "a");
original.put(new Point(3, 4), "b");
System.out.println(gson.toJson(original));
戻ります:
[
[
{
"x": 1,
"y": 2
},
"a"
],
[
{
"x": 3,
"y": 4
},
"b"
]
]
詳細は こちら をご覧ください。
GSONは、マップと複数のネストされたマップをシリアル化/逆シリアル化することを確信しています(つまり、Map<String, Map<String, Object>>
)デフォルトで問題ありません。あなたがもっと複雑な何かをする必要があるなら、私が信じる例は単なる出発点に過ぎないと信じています。
GSONソースのMapTypeAdapterFactoryクラスを確認してください: http://code.google.com/p/google-gson/source/browse/trunk/gson/src/main/Java/com/google/gson/ internal/bind/MapTypeAdapterFactory.Java
キーと値の型をJSON文字列にシリアル化できる限り(およびこれらのカスタムオブジェクト用に独自のシリアライザー/デシリアライザーを作成できる場合)、問題は発生しません。
Gson 2.7.2では、次のように簡単です
Gson gson = new Gson();
String serialized = gson.toJson(map);