JSON转换泛型对象
JSON 转含有泛型属性的对象
如果不知道干什么,那就学习吧
目录
正文
在将 json 字符串转为对象时,如果对象含有泛型,在进行转换时需要指明泛型类型。
1. 对象只含有一个泛型属性时
1.1 代码
/**
* @Describe:
* @Author: chenfan
* @Date: 2019/5/9 19:15 */@Data/**
* 含有泛型属性的对象 */class OneGeneric<E>{
E e;
String ss;
}
@Data/**
* 泛型对象 */class KeyMessage{
String kk;
}public class JsonTest { public static void main(String[] args) { // {"e": {"kk":"qwe"},"ss": "ssValue"}
String data = "{\"e\": {\"kk\":\"qwe\"},\"ss\": \"ssValue\"}";
Gson gson = new Gson(); // OneGeneric : 外层对象类型 KeyMessage :嵌套的泛型类型
OneGeneric one = gson.fromJson(data, new TypeToken<OneGeneric<KeyMessage>>() {}.getType());
System.out.println(one);
}
}1.2 运行结果
正确转换如下:

1.3 泛型属性为集合类型时
此时和只有一个泛型对象时处理方式一样
public class JsonTest { public static void main(String[] args) { // {"ss": "ssValue","list":[{"dd":"dd1"},{"dd":"dd2"}]}
String data = "{\"ss\": \"ssValue\",\"list\":[{\"dd\":\"dd1\"},{\"dd\":\"dd2\"}]}";
Gson gson = new Gson();
OneGeneric grandMessage = gson.fromJson(data, new TypeToken<OneGeneric<Generic>>() {}.getType());
System.out.println(grandMessage);
}
}/**
* 含有泛型属性的对象 */@Dataclass OneGeneric<E>{
List<E> list;
String ss;
}
@Dataclass Generic{
String dd;
}
2. 对象只含有多个泛型属性时
2.1 代码
/**
* @Describe:
* @Author: chenfan
* @Date: 2019/5/9 19:15 */@Data/**
* 含有多个泛型属性的对象 */class GrandMessage<K,V>{
K key;
V value;
}/**
* 泛型对象 */@Dataclass KeyMessage{
String kk;
}/**
* 泛型对象 */@Dataclass ValueMessage{
String vv;
}/**
* 测试类 */public class JsonTest { public static void main(String[] args) { // {"key": {"kk":"qwe"},"value": {"vv":"asd"}}
String data = "{\"key\": {\"kk\":\"qwe\"},\"value\": {\"vv\":\"asd\"}}";
Gson gson = new Gson();
GrandMessage grandMessage = gson.fromJson(data, new TypeToken<GrandMessage<KeyMessage,ValueMessage>>() {}.getType());
System.out.println(grandMessage);
}
}2.2 运行结果

2.3 泛型的顺序问题
在进行对象转换时, TypeToken<GrandMessage<KeyMessage,ValueMessage>> 中的泛型类型顺序必须按照 GrandMessage<K,V> 中声明的顺序,否则会对象属性全部为null
3. 泛型类型嵌套
3.1 代码
@Dataclass GrandMessage<K>{
K k;
String msg;
}
@Dataclass OneGeneric<E>{
E e;
String one;
}
@Dataclass Generic{
String dd;
}/**
* 测试类 */public class JsonTest { public static void main(String[] args) { // {"msg":"lili","k":{"one":"111","e":{"dd":"dddd"}}}
String data = "{\"msg\":\"lili\",\"k\":{\"one\":\"111\",\"e\":{\"dd\":\"dddd\"}}}";
Gson gson = new Gson(); // 转换时的泛型顺序一定要严格按照对象的嵌套顺序
GrandMessage grandMessage = gson.fromJson(data, new TypeToken<GrandMessage<OneGeneric<Generic>>>() {}.getType());
System.out.println(grandMessage);
}
}3.2 运行结果

4. 自定义转换逻辑
JSON转对象时,如果对象某个属性不确定(比如对象定义为Object,实际为 int 类型),需要自定义转换逻辑
显示指定 OpSessionControlMessage 的 messageId 为 int 类型:
private static Gson gson2 = new GsonBuilder()
.registerTypeAdapter( new TypeToken<OpSessionControlMessage>(){}.getType(), new JsonDeserializer<OpSessionControlMessage>() {
@Override public OpSessionControlMessage deserialize(
JsonElement json, Type typeOfT,
JsonDeserializationContext context) throws JsonParseException {
OpSessionControlMessage message = new OpSessionControlMessage();
JsonObject jsonObject = json.getAsJsonObject();
Set<Map.Entry<String, JsonElement>> entrySet = jsonObject.entrySet(); for (Map.Entry<String, JsonElement> entry : entrySet) {
Object ot = entry.getValue(); if(entry.getKey().equals("messageId")){
message.setOpType(((JsonPrimitive) ot).getAsInt());
}
} return message;
}
}).create();
