Проблема с GSON и экземпляромCreator

У меня есть следующие POJO:

public interface Shape {
    public double calcArea();
    public double calcPerimeter();
}

public class Rectangle implement Shape {
    // Various properties of a rectangle
}

public class Circle implements Shape {
    // Various properties of a circle
}

public class ShapeHolder {
    private List<Shape> shapes;

    // other stuff
}

У меня нет проблем с получением GSON для сериализации экземпляра ShapeHolder в JSON. Но когда я пытаюсь десериализовать строку этого JSON обратно в экземпляр ShapeHolder, я получаю ошибки:

String shapeHolderAsStr = getString();
ShapeHolder holder = gson.fromJson(shapeHodlderAsStr, ShapeHolder.class);

Броски:

Exception in thread "main" java.lang.RuntimeException: Unable to invoke no-args constructor for interface    
net.myapp.Shape. Register an InstanceCreator with Gson for this type may fix this problem.
    at com.google.gson.internal.ConstructorConstructor$8.construct(ConstructorConstructor.java:167)
    at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:162)
    ... rest of stack trace ommitted for brevity

Поэтому я посмотрел здесь и начал реализовывать свой собственный ShapeInstanceCreator:

public class ShapeInstanceCreator implements InstanceCreator<Shape> {
    @Override
    public Shape createInstance(Type type) {
        // TODO: ???
        return null;
    }
}

Но теперь я застрял: мне дан только java.lang.reflect.Type, но мне действительно нужен java.lang.Object чтобы я мог написать код:

public class ShapeInstanceCreator implements InstanceCreator<Shape> {
    @Override
    public Shape createInstance(Type type) {
        Object obj = convertTypeToObject(type);

        if(obj instanceof Rectangle) {
            Rectangle r = (Rectangle)obj;
            return r;
        } else {
            Circle c = (Circle)obj;
            return c;
        }

        return null;
    }
}

Что я могу сделать?

Обновить

В соответствии с предложением @raffian (ссылка, которую он/она разместил), я реализовал InterfaceAdapter точно так же, как в ссылке (я ничего не менял). Теперь я получаю следующее исключение:

Exception in thread "main" com.google.gson.JsonParseException: no 'type' member found in what was expected to be an interface wrapper
    at net.myapp.InterfaceAdapter.get(InterfaceAdapter.java:39)
    at net.myapp.InterfaceAdapter.deserialize(InterfaceAdapter.java:23)

Есть идеи?

Ответ 1

Вы просмотрели этот? Похож на хороший чистый способ реализации InstanceCreators.

Я тоже использовал Gson, но переключился на FlexJSON из-за проблем с сериализацией. С помощью Flex вам не нужны создатели экземпляров, просто убедитесь, что у ваших объектов есть getters/seters для всех полей на основе спецификации JavaBean, и вам хорошо идти:

 ShapeHolder sh = new ShapeHolder();
 sh.addShape(new Rectangle());
 sh.addShape(new Circle());
 JSONSerializer ser = new JSONSerializer();
 String json = ser.deepSerialize(sh);
 JSONDeserializer<ShapeHolder> der = new JSONDeserializer<ShapeHolder>();
 ShapeHolder sh2 = der.deserialize(json);

Ответ 2

ЗАМЕЧАНИЕ, что FlexJSON добавляет имя класса как часть json, как показано ниже.

{
    "HTTPStatus": "OK",
    "class": "com.XXX.YYY.HTTPViewResponse",
    "code": null,
    "outputContext": {
        "class": "com.XXX.YYY.ZZZ.OutputSuccessContext",
        "eligible": true
    }
}

Итак, JSON будет немного; но вам не нужно писать InstanceCreator, который требуется в GSON.