Конвертировать JSON в карту
Каков наилучший способ преобразовать код JSON следующим образом:
{
"data" :
{
"field1" : "value1",
"field2" : "value2"
}
}
в Java-карте, в которой используются ключи (field1, field2) и значения для этих полей (value1, value2).
Есть идеи? Должен ли я использовать Json-lib для этого? Или лучше, если я напишу свой собственный парсер?
20 ответов
Я надеюсь, что вы шутили о написании своего собственного парсера.:-)
Для такого простого отображения будет работать большинство инструментов из http://json.org/ (раздел java). Для одного из них (Джексон, http://wiki.fasterxml.com/JacksonInFiveMinutes) вы должны сделать:
HashMap<String,Object> result =
new ObjectMapper().readValue(JSON_SOURCE, HashMap.class);
(где JSON_SOURCE - это файл, поток ввода, читатель или строка содержимого json)
Использование библиотеки GSON:
import com.google.gson.Gson
import com.google.common.reflect.TypeToken
import java.lang.reclect.Type
Используйте следующий код:
Type mapType = new TypeToken<Map<String, Map>>(){}.getType();
Map<String, String[]> son = new Gson().fromJson(easyString, mapType);
Мне нравится библиотека Google GSON.
Когда вы не знаете структуру JSON. Ты можешь использовать
JsonElement root = new JsonParser().parse(jsonString);
и тогда вы можете работать с JSON. Например, как получить "value1" из вашего gson:
String value1 = root.getAsJsonObject().get("data").getAsJsonObject().get("field1").getAsString();
Используйте JSON lib Например, http://www.json.org/java/
// Assume you have a Map<String, String> in JSONObject jdata
@SuppressWarnings("unchecked")
Iterator<String> nameItr = jdata.keys();
Map<String, String> outMap = new HashMap<String, String>();
while(nameItr.hasNext()) {
String name = nameItr.next();
outMap.put(name, jdata.getString(name));
}
Мой пост может быть полезен для других, так что представьте, что у вас есть карта с определенным объектом в значениях, что-то вроде этого:
{
"shopping_list":{
"996386":{
"id":996386,
"label":"My 1st shopping list",
"current":true,
"nb_reference":6
},
"888540":{
"id":888540,
"label":"My 2nd shopping list",
"current":false,
"nb_reference":2
}
}
}
Для анализа этого файла JSON с библиотекой GSON легко: если ваш проект mavenized
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
<version>2.3.1</version>
</dependency>
Затем используйте этот фрагмент:
import com.google.gson.Gson;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
//Read the JSON file
JsonElement root = new JsonParser().parse(new FileReader("/path/to/the/json/file/in/your/file/system.json"));
//Get the content of the first map
JsonObject object = root.getAsJsonObject().get("shopping_list").getAsJsonObject();
//Iterate over this map
Gson gson = new Gson();
for (Entry<String, JsonElement> entry : object.entrySet()) {
ShoppingList shoppingList = gson.fromJson(entry.getValue(), ShoppingList.class);
System.out.println(shoppingList.getLabel());
}
Соответствующее POJO должно быть примерно таким:
public class ShoppingList {
int id;
String label;
boolean current;
int nb_reference;
//Setters & Getters !!!!!
}
Надеюсь, поможет!
С gson 2.7 от Google (возможно, и с более ранними версиями, но я тестировал 2.7) это так просто:
Map map = gson.fromJson(json, Map.class);
Который возвращает карту типа class com.google.gson.internal.LinkedTreeMap
и работает рекурсивно на вложенных объектах.
Я делаю это так. Это просто.
import java.util.Map;
import org.json.JSONObject;
import com.google.gson.Gson;
public class Main {
public static void main(String[] args) {
JSONObject jsonObj = new JSONObject("{ \"f1\":\"v1\"}");
@SuppressWarnings("unchecked")
Map<String, String> map = new Gson().fromJson(jsonObj.toString(),Map.class);
System.out.println(map);
}
}
Таким образом, он работает как карта...
JSONObject fieldsJson = new JSONObject(json);
String value = fieldsJson.getString(key);
<dependency>
<groupId>org.codehaus.jettison</groupId>
<artifactId>jettison</artifactId>
<version>1.1</version>
</dependency>
Если вы используете org.json, у JSONObject есть метод toMap()
. Вы легко можете сделать:
Map<String, Object> myMap = myJsonObject.toMap();
java.lang.reflect.Type mapType = new TypeToken<Map<String, Object>>(){}.getType();
Gson gson = new Gson();
Map<String, Object> categoryicons = gson.fromJson(json, mapType );
Попробуйте этот код:
public static Map<String, Object> convertJsonIntoMap(String jsonFile) {
Map<String, Object> map = new HashMap<>();
try {
ObjectMapper mapper = new ObjectMapper();
mapper.configure(DeserializationFeature.UNWRAP_ROOT_VALUE, true);
mapper.readValue(jsonFile, new TypeReference<Map<String, Object>>() {
});
map = mapper.readValue(jsonFile, new TypeReference<Map<String, String>>() {
});
} catch (IOException e) {
e.printStackTrace();
}
return map;
}
Библиотека JsonTools очень полная. Его можно найти на Github.
Еще одна альтернатива - json-simple, которую можно найти в Maven Central:
(JSONObject)JSONValue.parse(someString); //JSONObject is actually a Map.
Артефакт размером 24 Кбайт не имеет других зависимостей во время выполнения.
Если вам нужна чистая Java без каких-либо зависимостей, вы можете использовать сборку в Nashorn API из Java 8. Это устарело в Java 11.
Это работает для меня:
...
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;
...
public class JsonUtils {
public static Map parseJSON(String json) throws ScriptException {
ScriptEngineManager sem = new ScriptEngineManager();
ScriptEngine engine = sem.getEngineByName("javascript");
String script = "Java.asJSONCompatible(" + json + ")";
Object result = engine.eval(script);
return (Map) result;
}
}
Пример использования
JSON:
{
"data":[
{"id":1,"username":"bruce"},
{"id":2,"username":"clark"},
{"id":3,"username":"diana"}
]
}
Код:
...
import jdk.nashorn.internal.runtime.JSONListAdapter;
...
public static List<String> getUsernamesFromJson(Map json) {
List<String> result = new LinkedList<>();
JSONListAdapter data = (JSONListAdapter) json.get("data");
for(Object obj : data) {
Map map = (Map) obj;
result.add((String) map.get("username"));
}
return result;
}
JSON to Map всегда будет строковым / объектным типом данных. у меня есть GSON lib от Google. Библиотека Gson, работающая со строкой, а не со сложными объектами, вам нужно сделать что-то еще
Попробуйте этот кусок кода, у меня сработало,
Map<String, Object> retMap = new Gson().fromJson(
myJsonString, new TypeToken<HashMap<String, Object>>() {}.getType()
);
import net.sf.json.JSONObject
JSONObject.fromObject(yourJsonString).toMap
Библиотека Underscore-java может конвертировать строку json в хэш-карту. Я поддерживаю проект.
Пример кода:
import com.github.underscore.lodash.U;
import java.util.*;
public class Main {
@SuppressWarnings("unchecked")
public static void main(String[] args) {
String json = "{"
+ " \"data\" :"
+ " {"
+ " \"field1\" : \"value1\","
+ " \"field2\" : \"value2\""
+ " }"
+ "}";
Map<String, Object> data = (Map) U.get((Map<String, Object>) U.fromJson(json), "data");
System.out.println(data);
// {field1=value1, field2=value2}
}
}
Я знаю, что немного опаздываю с этим вопросом, но я разозлился и написал свой собственный парсер JSON. Который вы можете использовать для простого JSON из файла в карту.
Файл JSON:
{
"exercise": false,
"prove": "market",
"writing": true,
"eleven": [
99475723,
"view",
-862130728.7203984,
false,
false,
"pet"
],
"twelve": [
723,
"view",
-862130728.7203984,
false,
false,
"pet"
],
"damage": "characteristic",
"disappear": "mail"
}
Вывод карты:
exercise:false
prove:market
writing:true
eleven:[99475723, view, -8.621307287203984E8, false, false, pet]
twelve:[723, view, -8.621307287203984E8, false, false, pet]
damage:characteristic
disappear:mail
Я не рекомендую вам использовать его, если вы не понимаете его ограничений. Вот код.
Json.java
import java.io.*;
import java.util.ArrayList;
import java.util.Map;
public final class Json {
public static void saveJson(Map<String, Object> map, File file) {
StringBuilder sb = new StringBuilder();
sb.append('{');
if (map == null || map.size() == 0) {
sb.append('}');
} else {
int size = map.size(), at = 0;
for (Object o : map.keySet()) {
sb.append('"').append(o).append('"').append(':');
if (!(map.get(o) instanceof ArrayList)) {
Object x=map.get(o);
if (x instanceof Number || x instanceof Boolean) {
sb.append(x);
} else {
sb.append('"').append(x).append('"');
}
} else {
sb.append('[');
ArrayList<Object> list= (ArrayList<Object>) map.get(o);
int j=0,s=list.size();
for (Object value : list) {
if (value instanceof Number || value instanceof Boolean) {
sb.append(value);
} else {
sb.append('"').append(value).append('"');
}
if (s != ++j) {
sb.append(',');
}
}
sb.append(']');
}
sb.append((size != ++at) ? ',' : '}');
}
}
if (file != null)
try (BufferedWriter bw = new BufferedWriter(new FileWriter(file.getPath()),sb.length())) {
bw.append(sb);
bw.flush();
} catch (IOException e) {
e.printStackTrace();
}
}
public static boolean loadJson(File file,Map map) {
int read=0;
boolean exist = file.isFile();
if (exist) {
try (SimpleJsonFileReader sc = new SimpleJsonFileReader(file)) {
while (sc.hasNext) {
map.put(sc.nextString(), sc.nextObject());
read++;
}
} catch (IOException e) {
e.printStackTrace();
}
}
return exist && read>0;
}
}
SimpleJsonFileReader.java
import java.io.*;
import java.util.ArrayList;
public class SimpleJsonFileReader implements AutoCloseable {
final private int BUFFER_SIZE = 1 << 15;
private final DataInputStream din;
private final byte[] buffer;
private int bufferPointer, bytesRead;
public boolean hasNext;
protected long size = 0;
private boolean isInt;
public SimpleJsonFileReader(File file) throws IOException {
size = file.length();
hasNext = size > 2;
din = new DataInputStream(new FileInputStream(file.getPath()));
buffer = new byte[BUFFER_SIZE];
bufferPointer = bytesRead = 0;
}
public Object nextObject() throws IOException {
StringBuilder sb = next();
int n = sb.length();
boolean isNull = n < 1;
if (isNull) {
return null;
}
int c = sb.charAt(0);
boolean isString = (c == '"');
if (isString) {
return sb.substring(1, n - 1);
}
boolean isBoolean = (c == 't' || c == 'f');
if (isBoolean) {
return c == 't';
}
boolean isArray = (c == '[');
if (isArray) {
ArrayList<Object> arrayList = new ArrayList<>();
if (n > 1) {
arrayList.add(sb.substring(2, n - 1));
}
Object get = nextObject();
while (!(get instanceof StringBuilder)) {
arrayList.add(get);
get = nextObject();
}
StringBuilder last = (StringBuilder) get;
if (last.length() > 1) {
arrayList.add(last.substring(1, last.length() - 2));
}
return arrayList;
}
boolean isArrayEnd = (sb.charAt(sb.length() - 1) == ']');
if (isArrayEnd) {
return sb;
}
double x = nextDouble(sb);
if (isInt) {
isInt = false;
if (x > Integer.MAX_VALUE || x < Integer.MIN_VALUE) {
return (long) x;
} else {
return (int) x;
}
}
return x;
}
private StringBuilder next() throws IOException {
int n = read();
StringBuilder sb = new StringBuilder();
while (isWhiteSpace(n)) {
n = read();
}
while (!isWhiteSpace(n)) {
sb.append((char) n);
n = read();
}
hasNext = (size > 2);
return sb;
}
public String nextString() throws IOException {
int n = read();
StringBuilder sb = new StringBuilder();
while (isWhiteSpace(n)) {
n = read();
}
while (!isWhiteSpace(n)) {
sb.append((char) n);
n = read();
}
hasNext = (size > 1);
return sb.substring(1, sb.length() - 1);
}
public double nextDouble(StringBuilder sb) throws IOException {
double ret = 0, div = 1;
int i = 0, n = sb.length();
char c = sb.charAt(i);
boolean neg = (c == '-');
if (neg)
c = sb.charAt(++i);
do {
ret = ret * 10 + c - '0';
}
while (++i < n && (c = sb.charAt(i)) >= '0' && c <= '9');
isInt = !(c == '.');
if (!isInt) {
while (++i < n && (c = sb.charAt(i)) >= '0' && c <= '9') {
ret += (c - '0') / (div *= 10);
}
}
if (neg)
return -ret;
return ret;
}
private boolean isWhiteSpace(int n) {
return n == ' ' || n == '\n' || n == '\r' || n == '\t' || n == '{' || n == '}' || n == ',' || n == ':';
}
private void fillBuffer() throws IOException {
bytesRead = din.read(buffer, bufferPointer = 0, BUFFER_SIZE);
if (bytesRead == -1)
buffer[0] = -1;
}
private byte read() throws IOException {
size--;
if (bufferPointer == bytesRead)
fillBuffer();
return buffer[bufferPointer++];
}
@Override
public void close() throws IOException {
if (din == null)
return;
din.close();
}
}
Класс драйвера:test.java
import java.io.File;
import java.util.*;
public class Test {
public static void main(String[] args) {
LinkedHashMap linkedHashMap=new LinkedHashMap();
System.out.println(Json.loadJson(new File("gui.json"),linkedHashMap));;
for (Object o :linkedHashMap.keySet()) {
if(o instanceof Map){
Map map= (Map) o;
for (Object x : map.keySet()) {
System.out.println(x+":"+map.get(x));
}
}else {
System.out.println(o+":"+linkedHashMap.get(o));
}
}
}
}
JSON to Map всегда будет типом данных string/object. я получил GSON lib от Google.
работает очень хорошо и JDK 1.5 является минимальным требованием.