Ответ 1
используйте Jackson (http://jackson.codehaus.org/) из http://json.org/
HashMap<String,Object> result =
new ObjectMapper().readValue(<JSON_OBJECT>, HashMap.class);
У меня есть JSONObject
с некоторыми атрибутами, которые я хочу преобразовать в Map<String, Object>
Есть ли что-то, что я могу использовать с json.org или ObjectMapper
?
используйте Jackson (http://jackson.codehaus.org/) из http://json.org/
HashMap<String,Object> result =
new ObjectMapper().readValue(<JSON_OBJECT>, HashMap.class);
Вы можете использовать библиотеку Gson() (com.google.gson), если столкнетесь с какими-либо трудностями при использовании Джексона.
HashMap<String, Object> yourHashMap = new Gson().fromJson(yourJsonObject.toString(), HashMap.class);
Это то, что сработало для меня:
public static Map<String, Object> toMap(JSONObject jsonobj) throws JSONException {
Map<String, Object> map = new HashMap<String, Object>();
Iterator<String> keys = jsonobj.keys();
while(keys.hasNext()) {
String key = keys.next();
Object value = jsonobj.get(key);
if (value instanceof JSONArray) {
value = toList((JSONArray) value);
} else if (value instanceof JSONObject) {
value = toMap((JSONObject) value);
}
map.put(key, value);
} return map;
}
public static List<Object> toList(JSONArray array) throws JSONException {
List<Object> list = new ArrayList<Object>();
for(int i = 0; i < array.length(); i++) {
Object value = array.get(i);
if (value instanceof JSONArray) {
value = toList((JSONArray) value);
}
else if (value instanceof JSONObject) {
value = toMap((JSONObject) value);
}
list.add(value);
} return list;
}
Большая часть этого вопроса: Как преобразовать JSONObject в новую карту для всех ее ключей, используя итератор java
Обратите внимание на вышеупомянутое решение (от A Paul): Решение не работает, потому что оно не восстанавливает обратно HashMap < String, Object > - вместо этого он создает HashMap < String, LinkedHashMap > .
Причина в том, что во время демарширования каждый объект (JSON, маршаллированный как LinkedHashMap) используется как-есть, он принимает 1-на-1 LinkedHashMap (вместо преобразования LinkedHashMap обратно в соответствующий объект).
Если у вас есть HashMap < String, MyOwnObject > , то правильная демаршация была возможна - см. Следующий пример:
ObjectMapper mapper = new ObjectMapper();
TypeFactory typeFactory = mapper.getTypeFactory();
MapType mapType = typeFactory.constructMapType(HashMap.class, String.class, MyOwnObject.class);
HashMap<String, MyOwnObject> map = mapper.readValue(new StringReader(hashTable.toString()), mapType);
Обнаруженные эти проблемы можно решить, используя
ObjectMapper#convertValue(Object fromValue, Class<T> toValueType)
В результате исходное quuestion может быть решено в двухэтапном конверсировании:
Отмените JSON на объект, в котором Map<String, Object>
дешифруется как HashMap<String, LinkedHashMap>
, используя bjectMapper # readValue().
Преобразование внутренних ссылок LinkedHashMaps обратно в соответствующие объекты
ObjectMapper mapper = new ObjectMapper();
Class clazz = (Class) Class.forName(classType);
MyOwnObject value = mapper.convertValue(value, clazz);
Чтобы не допустить, чтобы "classType" должен был быть известен заранее, я принудительно применял при сортировке дополнительной карты, содержащей пары <key, classNameString>
. Таким образом, в момент unmarshalling классType может быть извлечен динамически.
Лучший способ преобразовать его в HashMap<String, Object>
заключается в следующем:
HashMap<String, Object> result = new ObjectMapper().readValue(jsonString, new TypeReference<Map<String, Object>>(){}));