Преобразование данных JSON в объект Java


262

Я хочу иметь возможность доступа к свойствам из строки JSON в моем методе действия Java. Строка доступна, просто сказав myJsonString = object.getJson(). Ниже приведен пример того, как может выглядеть строка:

{
    'title': 'ComputingandInformationsystems',
    'id': 1,
    'children': 'true',
    'groups': [{
        'title': 'LeveloneCIS',
        'id': 2,
        'children': 'true',
        'groups': [{
            'title': 'IntroToComputingandInternet',
            'id': 3,
            'children': 'false',
            'groups': []
        }]
    }]
}

В этой строке каждый объект JSON содержит массив других объектов JSON. Намерение состоит в том, чтобы извлечь список идентификаторов, где любой данный объект, обладающий свойством группы, содержит другие объекты JSON. Я смотрел на Gson от Google как на потенциальный плагин JSON. Может кто-нибудь предложить какую-либо форму руководства о том, как я могу генерировать Java из этой строки JSON?



Ответы:


329

Я смотрел на Gson от Google как на потенциальный плагин JSON. Может кто-нибудь предложить какую-либо форму руководства о том, как я могу генерировать Java из этой строки JSON?

Google Gson поддерживает дженерики и вложенные бины . []В JSON представляет собой массив , и должен отображаться в коллекции Java , такие как Listили просто обычный массив Java. {}В JSON представляет собой объект и должен отобразить на Java Mapили просто какой - то класс JavaBean.

У вас есть объект JSON с несколькими свойствами, groupsсвойство которого представляет собой массив вложенных объектов того же типа. Это можно проанализировать с помощью Gson следующим образом:

package com.stackoverflow.q1688099;

import java.util.List;
import com.google.gson.Gson;

public class Test {

    public static void main(String... args) throws Exception {
        String json = 
            "{"
                + "'title': 'Computing and Information systems',"
                + "'id' : 1,"
                + "'children' : 'true',"
                + "'groups' : [{"
                    + "'title' : 'Level one CIS',"
                    + "'id' : 2,"
                    + "'children' : 'true',"
                    + "'groups' : [{"
                        + "'title' : 'Intro To Computing and Internet',"
                        + "'id' : 3,"
                        + "'children': 'false',"
                        + "'groups':[]"
                    + "}]" 
                + "}]"
            + "}";

        // Now do the magic.
        Data data = new Gson().fromJson(json, Data.class);

        // Show it.
        System.out.println(data);
    }

}

class Data {
    private String title;
    private Long id;
    private Boolean children;
    private List<Data> groups;

    public String getTitle() { return title; }
    public Long getId() { return id; }
    public Boolean getChildren() { return children; }
    public List<Data> getGroups() { return groups; }

    public void setTitle(String title) { this.title = title; }
    public void setId(Long id) { this.id = id; }
    public void setChildren(Boolean children) { this.children = children; }
    public void setGroups(List<Data> groups) { this.groups = groups; }
    
    public String toString() {
        return String.format("title:%s,id:%d,children:%s,groups:%s", title, id, children, groups);
    }
}

Довольно просто, не правда ли? Просто имейте подходящий JavaBean и звоните Gson#fromJson().

Смотрите также:


4
Производительный? Вы на самом деле измерили это? Хотя GSON имеет разумный набор функций, я думал, что производительность была своего рода слабым местом (согласно [ cowtowncoder.com/blog/archives/2009/09/entry_326.html] ). Например: я думал, что GSON на самом деле не нужны сеттеры, и был основан на полях. Таким образом, код может быть немного упрощен.
StaxMan

3
Я использую это в приложении для Android. Это не самое быстрое решение, но его достаточно просто запрограммировать, чтобы оправдать отсутствие производительности для пользователя до сих пор. Возможно, в более поздней версии приложения оно будет удалено для более быстрого решения.
Януш

1
По скорости, если это достаточно быстро, это достаточно быстро. Я только что прокомментировал ссылку на ожидаемую хорошую производительность. С точки зрения набора функций, Джексон обрабатывает все те же вложения, наслоения и непатентованные объекты, так что разница не в скорости. Наличие геттеров и сеттеров никак не влияет на производительность (для пакетов, о которых я знаю), поэтому, безусловно, может иметь их там.
StaxMan

97
+1 для "пакета com.stackoverflow.q1688099;". По какой-то причине это заставило меня посмеяться.
GargantuChet

1
public String toString () {вернуть новый Gson (). toJson (this); // Используем это вместо записи каждой переменной}
Пракаш

45

Беваааааре Гсон! Это очень здорово, очень здорово, но второе вы хотите сделать что - нибудь кроме простых объектов, вы можете легко нужно начать строить свои собственные сериализаторы (который не является , что трудно).

Кроме того, если у вас есть массив объектов, и вы десериализуете некоторый JSON в этот массив объектов, истинные типы LOST! Полные объекты даже не будут скопированы! Используйте XStream. Который, используя jsondriver и устанавливая правильные настройки, будет кодировать уродливые типы в фактический json, так что вы ничего не потеряете. Небольшая цена (уродливый JSON) за настоящую сериализацию.

Обратите внимание, что Джексон исправляет эти проблемы и работает быстрее, чем GSON.


2
Я написал форк Gson, который исправляет эти проблемы (и избегает всех аннотаций Джексона): github.com/winterstein/flexi-gson
Даниэль Винтерштейн

26

Как ни странно, единственный приличный процессор JSON, упомянутый до сих пор, был GSON.

Вот более хороший выбор:

  • Jackson ( Github ) - мощная привязка данных (JSON к / из POJO), потоковая передача (ультрабыстрая), древовидная модель (удобная для нетипизированного доступа)
  • Flex-JSON - настраиваемая сериализация

РЕДАКТИРОВАТЬ (август / 2013):

Еще один вопрос для рассмотрения:

  • Genson - функциональность, аналогичная Джексону, предназначенная для упрощения настройки разработчиком

15

Или с Джексоном:

String json = "...
ObjectMapper m = new ObjectMapper();
Set<Product> products = m.readValue(json, new TypeReference<Set<Product>>() {});

Это даст ошибку, не может десериализовать экземпляр java.util.HashSet из токена
START_OBJECT

7

Если по каким-либо изменениям вы находитесь в приложении, которое уже использует http://restfb.com/, то вы можете сделать:

import com.restfb.json.JsonObject;

...

JsonObject json = new JsonObject(jsonString);
json.get("title");

и т.п.


Ваше решение короче и понятнее, почему оно получает только 3 отзыва? Что-то не так?
Джефф

4

Простой и рабочий код Java для преобразования JSONObjectвJava Object

Employee.java

import java.util.HashMap;
import java.util.Map;

import javax.annotation.Generated;

import com.fasterxml.jackson.annotation.JsonAnyGetter;
import com.fasterxml.jackson.annotation.JsonAnySetter;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.JsonPropertyOrder;

@JsonInclude(JsonInclude.Include.NON_NULL)
@Generated("org.jsonschema2pojo")
@JsonPropertyOrder({
"id",
"firstName",
"lastName"
})
public class Employee {

@JsonProperty("id")
private Integer id;
@JsonProperty("firstName")
private String firstName;
@JsonProperty("lastName")
private String lastName;
@JsonIgnore
private Map<String, Object> additionalProperties = new HashMap<String, Object>();

/**
*
* @return
* The id
*/
@JsonProperty("id")
public Integer getId() {
return id;
}

/**
*
* @param id
* The id
*/
@JsonProperty("id")
public void setId(Integer id) {
this.id = id;
}

/**
*
* @return
* The firstName
*/
@JsonProperty("firstName")
public String getFirstName() {
return firstName;
}

/**
*
* @param firstName
* The firstName
*/
@JsonProperty("firstName")
public void setFirstName(String firstName) {
this.firstName = firstName;
}

/**
*
* @return
* The lastName
*/
@JsonProperty("lastName")
public String getLastName() {
return lastName;
}

/**
*
* @param lastName
* The lastName
*/
@JsonProperty("lastName")
public void setLastName(String lastName) {
this.lastName = lastName;
}

@JsonAnyGetter
public Map<String, Object> getAdditionalProperties() {
return this.additionalProperties;
}

@JsonAnySetter
public void setAdditionalProperty(String name, Object value) {
this.additionalProperties.put(name, value);
}

}

LoadFromJSON.java

import org.codehaus.jettison.json.JSONObject;

import com.fasterxml.jackson.databind.ObjectMapper;

public class LoadFromJSON {

    public static void main(String args[]) throws Exception {
        JSONObject json = new JSONObject();
        json.put("id", 2);
        json.put("firstName", "hello");
        json.put("lastName", "world");

        byte[] jsonData = json.toString().getBytes();

        ObjectMapper mapper = new ObjectMapper();
        Employee employee = mapper.readValue(jsonData, Employee.class);

        System.out.print(employee.getLastName());

    }
}

Как получить доступ к этим свойствам JSON в JSP?
Zoran777

Если вы не хотите использовать его на странице JSP, код остается прежним. Класс Employee.java будет таким же, как и сейчас. Но код, написанный в LoadFromJSON.java, будет скопирован на страницу jsp с соответствующим импортом каждого класса. Отдых там без изменений вообще.
Рахул Райна


3

Если вы используете какие-либо специальные карты с ключами или значениями, а также специальные карты, вы обнаружите, что это не предусмотрено реализацией Google.


2

Что не так со стандартными вещами?

JSONObject jsonObject = new JSONObject(someJsonString);
JSONArray jsonArray = jsonObject.getJSONArray("someJsonArray");
String value = jsonArray.optJSONObject(i).getString("someJsonValue");

это ужасно медленно: недавно на работе github.com/fabienrenaud/java-json-benchmark я удвоил производительность (сокращение использования ЦП в два раза, задержка сократилась), переключив все вызовы org.json ser / deserialization использовать Джексона.
Фабьен

0

Дайте Бун попробовать:

https://github.com/RichardHightower/boon

Это злой быстро

https://github.com/RichardHightower/json-parsers-benchmark

Не верьте мне на слово ... проверьте тест Гатлинга.

https://github.com/gatling/json-parsers-benchmark

(В некоторых случаях до 4х и из сотен тестов. У него также есть режим наложения индекса, который еще быстрее. Он молодой, но уже имеет некоторых пользователей.)

Он может анализировать JSON в Maps и Lists быстрее, чем любая другая библиотека может анализировать в JSON DOM, и это без режима наложения индекса. С режимом наложения индекса Boon, это еще быстрее.

Он также имеет очень быстрый режим JSON lax и режим парсера PLIST. :) (и имеет очень мало памяти, напрямую из байтового режима с кодировкой UTF-8 на лету).

Он также имеет самый быстрый режим JSON в JavaBean.

Это новое, но если вы ищете скорость и простой API, я не думаю, что есть более быстрый или более минималистичный API.


Можете ли вы предоставить ссылку на актуальную документацию для последней версии? На сегодняшний день я нашел 0.4, но не могу легко найти соответствующую ссылку на документ или учебник для этой версии. Спасибо
Бизмарк

Вот учебник github.com/RichardHightower/boon/wiki/Boon-JSON-in-five-minutes Бун находится в публичном репозитории Maven. Это на 0,27 или около того.
RickHigh

У richardhightower.github.io/site/releases есть 0,4, так что я подумал, что это было последним. Я проверял Boon для проекта на работе, имеет ли он эквивалентную аннотацию к @JsonIgnore Джексона?
Бизмарк

boon находится на нижнем уровне производительности ser / deserialization: github.com/fabienrenaud/java-json-benchmark Выберите jackson или dsljson для производительности. @RickHigh: Я не смог открыть проблему на вашем github, я более чем готов улучшить свой тест или исправить его интерпретацию, если что-то не так / что я пропустил.
Фабьен

0

В зависимости от входного формата JSON (строка / файл) создайте jSONString. Пример объекта класса Message, соответствующего JSON, можно получить, как показано ниже:

Сообщение msgFromJSON = new ObjectMapper (). ReadValue (jSONString, Message.class);


0

Самый простой способ заключается в том, что вы можете использовать этот метод softconvertvalue, который является пользовательским методом, в котором вы можете конвертировать jsonData в ваш определенный класс Dto.

Dto response = softConvertValue(jsonData, Dto.class);


public static <T> T softConvertValue(Object fromValue, Class<T> toValueType) 
{
    ObjectMapper objMapper = new ObjectMapper();
    return objMapper
        .configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
        .convertValue(fromValue, toValueType);
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.