Java - 使用 Gson 序列化 Iterable<Map.Entry<>>
Java - Serializing Iterable<Map.Entry<>> with Gson
我尝试使用 google GSON 库从类型 Map.Entry 序列化一个 "Iterable" -
我得到一个空输出。
这里是代码示例:
static private Iterable<Map.Entry<String, Integer>> getIterable(){
HashMap<String, Integer> map = new HashMap<>();
map.put("1", 1);
map.put("2", 2);
Iterable<Map.Entry<String, Integer>> iterable = map.entrySet();
return iterable;
}
public static void main(String[] args) {
Iterable<Map.Entry<String, Integer>> myIterable = getIterable();
GsonBuilder builder = new GsonBuilder();
Gson gson = builder.enableComplexMapKeySerialization().create();
String a= gson.toJson(myIterable);
System.out.println(a);
}
这是输出:
[{},{}]
知道我做错了什么吗?
谢谢 :)
java版本:1.8
gson 版本:2.6.2
更简洁的方法可能是
final Iterable<Entry<String, Integer>> iterable = getIterable();
final Gson gson = new Gson();
final JsonArray jsonArray = new JsonArray();
for (final Entry<String, Integer> entry : iterable) {
final JsonElement jsonElement = gson.toJsonTree(entry);
jsonElement.getAsJsonObject().remove("hash");
jsonArray.add(jsonElement);
}
或者我喜欢的 Stream
版本
StreamSupport.stream(iterable.spliterator(), false)
.map(gson::toJsonTree)
.map(JsonElement::getAsJsonObject)
.peek(obj -> obj.remove("hash"))
.collect(of(
JsonArray::new,
(array, obj) -> array.add(obj),
(output, toMerge) -> {
output.addAll(toMerge);
return output;
}
));
输出:[{"key":"1","value":1},{"key":"2","value":2}]
TL;DR:您需要一个自定义 TypeAdapterFactory
和 一个自定义 TypeAdapter
.
在 TypeAdapters
上查看此方法
public static <TT> TypeAdapterFactory newFactory(
final TypeToken<TT> type, final TypeAdapter<TT> typeAdapter) {
return new TypeAdapterFactory() {
@SuppressWarnings("unchecked") // we use a runtime check to make sure the 'T's equal
@Override public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> typeToken) {
return typeToken.equals(type) ? (TypeAdapter<T>) typeAdapter : null;
}
};
}
没有自定义 TypeAdapterFactory
typeToken.equals(type)
returnsfalse
,连自定义的TypeAdapter<Entry>
都不用。
问题出在这里,在 ReflectiveTypeAdapterFactory#write
@Override public void write(JsonWriter out, T value) throws IOException {
if (value == null) {
out.nullValue();
return;
}
out.beginObject();
try {
for (BoundField boundField : boundFields.values()) {
if (boundField.writeField(value)) {
out.name(boundField.name);
boundField.write(out, value);
}
}
} catch (IllegalAccessException e) {
throw new AssertionError(e);
}
out.endObject();
}
并在 ReflectiveTypeAdapterFactory#getBoundFields
private Map<String, BoundField> getBoundFields(Gson context, TypeToken<?> type, Class<?> raw) {
Map<String, BoundField> result = new LinkedHashMap<String, BoundField>();
if (raw.isInterface()) {
return result;
}
Gson 将输入 Entry
(Class<?> raw
参数)识别为
interface Map.Entry<K, V> { ... }
因此
if (raw.isInterface())
yield true
,并返回一个空 boundFields
LinkedHashMap
。
因此,这里
for (BoundField boundField : boundFields.values()) { ... }
循环没有执行,没有提取和写入值
boundField.write(...)
我尝试使用 google GSON 库从类型 Map.Entry 序列化一个 "Iterable" - 我得到一个空输出。
这里是代码示例:
static private Iterable<Map.Entry<String, Integer>> getIterable(){
HashMap<String, Integer> map = new HashMap<>();
map.put("1", 1);
map.put("2", 2);
Iterable<Map.Entry<String, Integer>> iterable = map.entrySet();
return iterable;
}
public static void main(String[] args) {
Iterable<Map.Entry<String, Integer>> myIterable = getIterable();
GsonBuilder builder = new GsonBuilder();
Gson gson = builder.enableComplexMapKeySerialization().create();
String a= gson.toJson(myIterable);
System.out.println(a);
}
这是输出:
[{},{}]
知道我做错了什么吗?
谢谢 :)
java版本:1.8
gson 版本:2.6.2
更简洁的方法可能是
final Iterable<Entry<String, Integer>> iterable = getIterable();
final Gson gson = new Gson();
final JsonArray jsonArray = new JsonArray();
for (final Entry<String, Integer> entry : iterable) {
final JsonElement jsonElement = gson.toJsonTree(entry);
jsonElement.getAsJsonObject().remove("hash");
jsonArray.add(jsonElement);
}
或者我喜欢的 Stream
版本
StreamSupport.stream(iterable.spliterator(), false)
.map(gson::toJsonTree)
.map(JsonElement::getAsJsonObject)
.peek(obj -> obj.remove("hash"))
.collect(of(
JsonArray::new,
(array, obj) -> array.add(obj),
(output, toMerge) -> {
output.addAll(toMerge);
return output;
}
));
输出:[{"key":"1","value":1},{"key":"2","value":2}]
TL;DR:您需要一个自定义 TypeAdapterFactory
和 一个自定义 TypeAdapter
.
在 TypeAdapters
public static <TT> TypeAdapterFactory newFactory(
final TypeToken<TT> type, final TypeAdapter<TT> typeAdapter) {
return new TypeAdapterFactory() {
@SuppressWarnings("unchecked") // we use a runtime check to make sure the 'T's equal
@Override public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> typeToken) {
return typeToken.equals(type) ? (TypeAdapter<T>) typeAdapter : null;
}
};
}
没有自定义 TypeAdapterFactory
typeToken.equals(type)
returnsfalse
,连自定义的TypeAdapter<Entry>
都不用。
问题出在这里,在 ReflectiveTypeAdapterFactory#write
@Override public void write(JsonWriter out, T value) throws IOException {
if (value == null) {
out.nullValue();
return;
}
out.beginObject();
try {
for (BoundField boundField : boundFields.values()) {
if (boundField.writeField(value)) {
out.name(boundField.name);
boundField.write(out, value);
}
}
} catch (IllegalAccessException e) {
throw new AssertionError(e);
}
out.endObject();
}
并在 ReflectiveTypeAdapterFactory#getBoundFields
private Map<String, BoundField> getBoundFields(Gson context, TypeToken<?> type, Class<?> raw) {
Map<String, BoundField> result = new LinkedHashMap<String, BoundField>();
if (raw.isInterface()) {
return result;
}
Gson 将输入 Entry
(Class<?> raw
参数)识别为
interface Map.Entry<K, V> { ... }
因此
if (raw.isInterface())
yield true
,并返回一个空 boundFields
LinkedHashMap
。
因此,这里
for (BoundField boundField : boundFields.values()) { ... }
循环没有执行,没有提取和写入值
boundField.write(...)