我正在尝试弄清如何反序列化EnumMap。到目前为止,我一直在使用Gson库进行所有其他操作,并且取得了成功。但这证明是困难的。
下面是一个基本思路:
import java.lang.reflect.Type;
import com.google.gson.reflect.TypeToken;
import com.google.gson.Gson;
enum FRUIT {
APPLE, BANANA
}
EnumMap<FRUIT, String> fruitMap;
Gson gson = new Gson();
public void setFruitMap(String jsonString){
Type typeToken = new TypeToken<EnumMap<FRUIT, String>>(){}.getType();
fruitMap = gson.fromJson(jsonString, typeToken);
}
String fruitMapString = "{ \"BANANA\":\"tasty!\", \"APPLE\":\"gross!\" }";
setFruitMap(fruitMapString); //Error occurs here.
assertEquals("tasty!", fruitMap.get(FRUIT.BANANA));
当我运行上面的代码时,我得到一个java.lang.ClassCastException: java.util.LinkedHashMap 无法转换为java.util.EnumMap的错误提示。
所以在我看来,Gson库没有创建EnumMap, 而是在制作LinkedHashMap之后尝试进行转换。
因此,我想我会去编写自己的反序列化逻辑。这里是一个实现,它可以工作,但是......它有点不稳定。
public JsonDeserializer<EnumMap<FRUIT, String>> deserializeFruitMap(){
return new JsonDeserializer<EnumMap<FRUIT, String>>(){
@Override
public EnumMap<FRUIT, String> deserialize(JsonElement element, Type typeOf, JsonDerializationContext context){
Type token = new TypeToken<HashMap<String, String>>(){}.getType();
HashMap<String, String> tempMap = context.deserialize(element, token);
EnumMap<FRUIT, String> fruitMap = new EnumMap<>(FRUIT.class);
for(Entry<String, String> entry : tempMap.entrySet){
fruitMap.put(FRUIT.valueOf(entry.getKey()), entry.getValue());
}
return fruitMap;
}
};
}
虽然它能正常工作,但不够美观。而且很具体化。我真的希望将其抽象为类似于......
public <K extends Enum<K>, V> JsonDeserializer<EnumMap<K, V>> deserializeEnumMap(){
return new JsonDeserializer<EnumMap<K, V>>(){
@Override
public EnumMap<K, V> deserialize(JsonElement element, Type typeOf, JsonDerializationContext context){
Type token = new TypeToken<HashMap<String, String>>(){}.getType();
HashMap<String, String> tempMap = context.deserialize(element, token);
EnumMap<K, V> enumMap = new EnumMap<>(K.class); //This doesn't work.
for(Entry<String, String> entry : tempMap.entrySet){
fruitMap.put(K.valueOf(entry.getKey()), entry.getValue());
}
return enumMap;
}
};
}
请问有谁知道以下问题的解决办法吗? 1)如何改进第一种方法? 2)如何使抽象版本起作用? 谢谢!
我不得不在方法中添加一个转换,但它确实有效。
fruitMap = newEnumMap<FRUIT, String>((Map<FRUIT, String>) gson.fromJson(jsonString, typeToken));
- C Taylor