Конвертировать 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 является минимальным требованием.

Другие вопросы по тегам