Gson не сериализует поля, определенные в подклассах - PullRequest
5 голосов
/ 16 ноября 2011

По неизвестной причине, если у меня есть:

class A{
int stars;
public int getStars(){
return stars;
}

public void setStarts(int stars){
this.stars = stars;
}
}

class B extends A{
 int sunshines;

[getter and setter for sunshines]
}

class C{
 List<A> classes;
[get and set for classes]
}

если я сериализую Объект типа C, у меня будут только поля A в сериализованных объектах в полевых классах (в то время как я ожидал бы иметь поля B, если объект является B).

Как это сделать?

Ответы [ 4 ]

9 голосов
/ 11 июня 2013

Gson 2.1 поддерживает это из коробки:

import java.util.ArrayList;
import java.util.List;

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;

public class GsonInheritanceTest
{
  public static void main(String[] args)
  {
    Gson gson = new GsonBuilder().setPrettyPrinting().create();
    C c = new C();
    c.classes = new ArrayList<A>();
    c.classes.add(new A(1));
    c.classes.add(new B(2, 3));
    System.out.println(gson.toJson(c));
  }

  static class A
  {
    int stars;

    A(int stars)
    {
      this.stars = stars;
    }
  }

  static class B extends A
  {
    int sunshines;

    B(int stars, int sunshines)
    {
      super(stars);
      this.sunshines = sunshines;
    }
  }

  static class C
  {
    List<A> classes;
  }
}

Выход составляет

{
  "classes": [
    {
      "stars": 1
    },
    {
      "sunshines": 3,
      "stars": 2
    }
  ]
}
3 голосов
/ 22 февраля 2013

Я использовал ответ Ивана, чтобы подсказать мне более элегантное решение - оно не идеальное, но оно работало достаточно хорошо для меня. Я создал новый JsonSerializer, который выбрасывает тип A, приводя его к объекту. Кажется, что при задании простого объекта Gson смотрит на фактический тип объекта, а не ссылается на тип передаваемой переменной.

public static class ASerializer implements JsonSerializer<A> {

    @Override
    public JsonElement serialize( A in, Type type, JsonSerializationContext ctx ) {
        return ctx.serialize( (Object) in );
    }
}

Затем вы передаете ASerializer GsonBuilder, чтобы получить экземпляр Gson для использования:

GsonBuilder gb = new GsonBuilder();
gb.registerTypeAdapter( A.class, new ASerializer() ); 
Gson gson = gb.create();
1 голос
/ 20 ноября 2011

Поддержка Gson для обработки полиморфных типов неполная.

Если возможно, я настоятельно рекомендую переключиться на Джексона , что решает проблему сериализации в исходном вопросе без ошибок.

Если необходимо использовать Gson, то необходима пользовательская обработка сериализации для обработки такой структуры данных. К счастью, реализация пользовательского сериализатора может быть довольно простой. Например:

import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonElement;
import com.google.gson.JsonSerializationContext;
import com.google.gson.JsonSerializer;

public class GsonFoo
{
  public static void main(String[] args)
  {
    C c = new C();
    c.classes = new ArrayList<A>();
    c.classes.add(new A(42));
    c.classes.add(new B(8));

    System.out.println(new Gson().toJson(c));
    // output:
    // {"classes":[{"stars":42},{"stars":-1}]}

    GsonBuilder gsonBuilder = new GsonBuilder();
    gsonBuilder.registerTypeAdapter(A.class, new ASerializer());
    Gson gson = gsonBuilder.create();
    System.out.println(gson.toJson(c));
    // output:
    // {"classes":[{"stars":42},{"sunshines":8,"stars":-1}]}
  }
}

class ASerializer implements JsonSerializer<A>
{
  @Override
  public JsonElement serialize(A src, Type typeOfSrc, JsonSerializationContext context)
  {
    Gson gson = new Gson();
    return gson.toJsonTree(src);
  }
}

class A
{
  public int stars;

  A(int s)
  {
    stars = s;
  }
}

class B extends A
{
  public int sunshines;

  B(int s)
  {
    super(-1);
    sunshines = s;
  }
}

class C
{
  public List<A> classes;
}
1 голос
/ 16 ноября 2011

Я попробовал ваш пример, и замена List<A> на List<Object> в классе C решила эту проблему.

import java.util.ArrayList;
import java.util.List;

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;

public class GsonInheritanceTest {
public static void main(String[] args) {
    Gson gson = new GsonBuilder().setPrettyPrinting().create();
    C c = new C();
    c.classes = new ArrayList<Object>();
    c.classes.add(new A());
    c.classes.add(new B());
    System.out.println(gson.toJson(c));
}

public static class A {
    int stars;
}

public static class B extends A {
    int sunshines;
}

public static class C {
    List<Object> classes;
}
}

Производит:

{
  "classes": [
    {
      "stars": 0
    },
    {
      "sunshines": 0,
      "stars": 0
    }
  ]
}

Это, однако, не позволяет использовать десериализацию по умолчанию json ... Рассмотрите возможность рефакторинга OOD и отношений наследования.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...