Как разобрать массив JSON с помощью Gson

86

Я хочу анализировать массивы JSON и использовать gson. Во-первых, я могу регистрировать вывод JSON, сервер четко реагирует на клиента.

Вот мой вывод JSON:

 [
      {
           id : '1',
           title: 'sample title',
           ....
      },
      {
           id : '2',
           title: 'sample title',
           ....
     },
      ...
 ]

Я пробовал эту структуру для разбора. Класс, зависящий от единственного arrayи ArrayListдля всех JSONArray.

 public class PostEntity {

      private ArrayList<Post> postList = new ArrayList<Post>();

      public List<Post> getPostList() { 
           return postList; 
      }

      public void setPostList(List<Post> postList) { 
           this.postList = (ArrayList<Post>)postList; 
      } 
 }

Почтовый класс:

 public class Post {

      private String id;
      private String title;

      /* getters & setters */
 }

Когда я пытаюсь использовать gson без ошибок, без предупреждений и без журнала:

 GsonBuilder gsonb = new GsonBuilder();
 Gson gson = gsonb.create();

 PostEntity postEnt;
 JSONObject jsonObj = new JSONObject(jsonOutput);
 postEnt = gson.fromJson(jsonObj.toString(), PostEntity.class);

 Log.d("postLog", postEnt.getPostList().get(0).getId());

Что не так, как решить?

Огулкан Орхан
источник

Ответы:

251

Вы можете разобрать JSONArrayнапрямую, не нужно еще раз оборачивать свой Postкласс PostEntityи не нуждаться в новом JSONObject().toString():

Gson gson = new Gson();
String jsonOutput = "Your JSON String";
Type listType = new TypeToken<List<Post>>(){}.getType();
List<Post> posts = gson.fromJson(jsonOutput, listType);

Надеюсь, это поможет.

Йорк
источник
Я удалил класс PostEntity и вместо этого попробовал ваш фрагмент. По-прежнему никаких изменений. Благодарю.
Огулкан Орхан
Наконец то сработало правильно и качественно. Большое вам спасибо еще раз.
Огулкан Орхан
Привет, как разбирать данные вложенного массива json, как это .... [{"firstName": "Bidhan", "lastName": "Chatterjee"}, [{"type": "personal", "number": "09832209761" }, {"тип": "факс", "номер": "91-342-2567692"}]]
KK_07k11A0585
Есть ли польза от использования TypeToken? Судя по моим тестам, похоже, что gson обрабатывает поле List <Object> без использования TypeToken.
greg7gkb
Это действительно полезно для меня
bhavesh kaila
6

Я искал способ более общего анализа массивов объектов; вот мой вклад:

CollectionDeserializer.java:

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;

import com.google.gson.Gson;
import com.google.gson.JsonArray;
import com.google.gson.JsonDeserializationContext;
import com.google.gson.JsonDeserializer;
import com.google.gson.JsonElement;
import com.google.gson.JsonParseException;

public class CollectionDeserializer implements JsonDeserializer<Collection<?>> {

    @Override
    public Collection<?> deserialize(JsonElement json, Type typeOfT,
            JsonDeserializationContext context) throws JsonParseException {
        Type realType = ((ParameterizedType)typeOfT).getActualTypeArguments()[0];

        return parseAsArrayList(json, realType);
    }

    /**
     * @param serializedData
     * @param type
     * @return
     */
    @SuppressWarnings("unchecked")
    public <T> ArrayList<T> parseAsArrayList(JsonElement json, T type) {
        ArrayList<T> newArray = new ArrayList<T>();
        Gson gson = new Gson();

        JsonArray array= json.getAsJsonArray();
        Iterator<JsonElement> iterator = array.iterator();

        while(iterator.hasNext()){
            JsonElement json2 = (JsonElement)iterator.next();
            T object = (T) gson.fromJson(json2, (Class<?>)type);
            newArray.add(object);
        }

        return newArray;
    }

}

JSONParsingTest.java:

public class JSONParsingTest {

    List<World> worlds;

    @Test
    public void grantThatDeserializerWorksAndParseObjectArrays(){

        String worldAsString = "{\"worlds\": [" +
            "{\"name\":\"name1\",\"id\":1}," +
            "{\"name\":\"name2\",\"id\":2}," +
            "{\"name\":\"name3\",\"id\":3}" +
        "]}";

        GsonBuilder builder = new GsonBuilder();
        builder.registerTypeAdapter(Collection.class, new CollectionDeserializer());
        Gson gson = builder.create();
        Object decoded = gson.fromJson((String)worldAsString, JSONParsingTest.class);

        assertNotNull(decoded);
        assertTrue(JSONParsingTest.class.isInstance(decoded));

        JSONParsingTest decodedObject = (JSONParsingTest)decoded;
        assertEquals(3, decodedObject.worlds.size());
        assertEquals((Long)2L, decodedObject.worlds.get(1).getId());
    }
}

World.java:

public class World {
    private String name;
    private Long id;

    public void setName(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

}
Miere
источник
@Miere Я не понимаю, как это конвертируется из JsonArray в List <World>
ARK
По сути, он использует CollectionDeserializer для каждой коллекции, обнаруженной в процессе десериализации GSON. CollectionDeserializer, в свою очередь, определяет через общий параметр Collection, какой класс представляет хранилище объектов в коллекции, которую он должен десериализовать. Надеюсь, это ответ на ваш вопрос. Если у вас все еще есть сомнения, свяжитесь со мной по адресу [miere00 at gmail.com].
Miere
Привет @Miere, это решение не решает исходный вопрос. Ваш JSONParsingTest - это объект с массивом внутри. Как вы используете свой CollectionDeserializer для анализа только массива, а не массива, заключенного в объект?
zkon
5

Преобразовать в массив объектов

Gson gson=new Gson();
ElementType [] refVar=gson.fromJson(jsonString,ElementType[].class);

Преобразовать как тип сообщения

Gson gson=new Gson();
Post [] refVar=gson.fromJson(jsonString,Post[].class);

Чтобы прочитать его как список объектов, можно использовать TypeToken

List<Post> posts=(List<Post>)gson.fromJson(jsonString, 
                     new TypeToken<List<Post>>(){}.getType());
xrcwrn
источник
как насчет ArrayList <Post>?
гумурух
3
Type listType = new TypeToken<List<Post>>() {}.getType();
List<Post> posts = new Gson().fromJson(jsonOutput.toString(), listType);
Салман Хан
источник
6
Хотя это может дать ответ на вопрос, это не дает никакого контекста, чтобы объяснить, как и почему. Вы можете добавить одно или два предложения, чтобы объяснить свой ответ.
brandonscript
3

Некоторые ответы в этом посте действительны, но с использованием TypeToken библиотека Gson генерирует объекты Tree с нереальными типами для вашего приложения.

Чтобы получить это, мне пришлось прочитать массив и преобразовать один за другим объекты внутри массива. Конечно, этот метод не самый быстрый, и я не рекомендую его использовать, если у вас массив слишком большой, но у меня он сработал.

В проект необходимо включить библиотеку Json. Если вы разрабатываете на Android, в него входят:

/**
 * Convert JSON string to a list of objects
 * @param sJson String sJson to be converted
 * @param tClass Class
 * @return List<T> list of objects generated or null if there was an error
 */
public static <T> List<T> convertFromJsonArray(String sJson, Class<T> tClass){

    try{
        Gson gson = new Gson();
        List<T> listObjects = new ArrayList<>();

        //read each object of array with Json library
        JSONArray jsonArray = new JSONArray(sJson);
        for(int i=0; i<jsonArray.length(); i++){

            //get the object
            JSONObject jsonObject = jsonArray.getJSONObject(i);

            //get string of object from Json library to convert it to real object with Gson library
            listObjects.add(gson.fromJson(jsonObject.toString(), tClass));
        }

        //return list with all generated objects
        return listObjects;

    }catch(Exception e){
        e.printStackTrace();
    }

    //error: return null
    return null;
}
JSR - Silicornio
источник
3
[
      {
           id : '1',
           title: 'sample title',
           ....
      },
      {
           id : '2',
           title: 'sample title',
           ....
     },
      ...
 ]

Проверьте простой код для этого вывода

 Gson gson=new GsonBuilder().create();
                List<Post> list= Arrays.asList(gson.fromJson(yourResponse.toString,Post[].class));
Прабх глубокий
источник
1

Вы можете легко сделать это в Kotlin, используя следующий код:

val fileData = "your_json_string"
val gson = GsonBuilder().create()
val packagesArray = gson.fromJson(fileData , Array<YourClass>::class.java).toList()

В принципе, вам нужно только , чтобы обеспечить Arrayиз YourClassобъектов.

xarlymg89
источник
Почему вы используете PrettyPrinting?
Antroid
@antroid Я забыл удалить его ^^
xarlymg89
0

вы можете получить значение списка без использования объекта Type .

EvalClassName[] evalClassName;
ArrayList<EvalClassName> list;
evalClassName= new Gson().fromJson(JSONArrayValue.toString(),EvalClassName[].class);
list = new ArrayList<>(Arrays.asList(evalClassName));

Я протестировал его, и он работает.

Сумит Кумар
источник
0

в Котлине :

val jsonArrayString = "['A','B','C']"

val gson = Gson()

val listType: Type = object : TypeToken<List<String?>?>() {}.getType()

val stringList : List<String> = gson.fromJson(
                            jsonArrayString,
                            listType)
Аднан Абдолла Заки
источник