2

我有一个 json 数据包,由一个 dacket 标头(在 PACKET_HEADER 下)+许多不同类型的消息(在 DATA 下)组成。这些消息具有相同的父类:

//my packet class
public class Packet {
    public PacketHeader PACKET_HEADER;
    public Message[] DATA;
}

//my parent message class:
public class Message {

    public enum MessageType {
        WHOAREYOU,
        TimeData,
        Ready,
        LocationData
    }

    public MessageType getMessageType() {
        return MESSAGE_TYPE;
    }

    public void setMessageType(MessageType MessageType) {
        this.MESSAGE_TYPE = MessageType;
    }

    private MessageType MESSAGE_TYPE;
    //declaring public for simplicity
    public String SENDER;
    public String SENDER_TYPE;

}

//sample child class:
public class TimeMessage extends Message {

    private int tick;
    public int getTick()
    {
        return tick;
    }

我像这样反序列化输入:

 @Override
    public Handler create(Connector connector, Object packet, int clientID) {
        Gson gson = new Gson();
        Packet pack = gson.fromJson(packet.toString(), Packet.class);
        System.out.println("Number of messages : " + pack.PACKET_HEADER.NOF_MESSAGES );//ok
        for(Message msg : pack.DATA)
        {
            switch (msg.getMessageType()) {
                //enters the switch successfully
                case TimeData: {               
                    System.out.println("Sender:  " + msg.SENDER  + "  Sender_type: " + msg.SENDER_TYPE);//ok
                    TimeMessage time_msg = (TimeMessage) msg;//NOT ok, generates exception!!!!
                    //...
                    return new TimeHandler(time_msg, connector, clientID);
                }
                default:                    
                    System.out.println("Switch " + msg.getMessageType() + " is invalid");
                    return null;
            }
        }
       return null; 

    }

你知道为什么我不能投到儿童班吗?我该如何解决。提前非常感谢。

4

1 回答 1

4

这是因为 Gson 在没有一点帮助的情况下无法处理多态性。

实际上JSON 不保留类型信息,因此当 Gson 反序列化文档时,它只会创建Message 实例,而不是TimeMessage等子类的实例,因为它无法猜测正确的类型。

您必须使用附加组件RuntimeTypeAdapterFactory

我刚刚在我的博客上写了一篇关于它的文章:http: //pragmateek.com/javajson-mapping-with-gson

这是您感兴趣的部分:http: //pragmateek.com/javajson-mapping-with-gson/#Preserving_type_information

请问您是否还有问题。

编辑:

这是针对您的具体情况的完整示例:

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

public class Program
{
    public static class Packet
    {
        public Message[] DATA;
    }

    public static enum MessageType
    {
        WHOAREYOU,
        TimeData,
        Ready,
        LocationData
    }

    //my parent message class:
    public static class Message {

        public MessageType getMessageType() {
            return MESSAGE_TYPE;
        }

        public void setMessageType(MessageType MessageType) {
            this.MESSAGE_TYPE = MessageType;
        }

        private MessageType MESSAGE_TYPE;
        //declaring public for simplicity
        public String SENDER;
        public String SENDER_TYPE;

    }

    //sample child class:
    public static class TimeMessage extends Message
    {
        private int tick;
        public int getTick()
        {
            return tick;
        }
        public TimeMessage(int tick)
        {
            this.tick = tick;
        }
    }

    public static void main(String[] args)
    {
        Message[] messages = { new TimeMessage(123) };

        GsonBuilder builder = new GsonBuilder();
        builder.registerTypeAdapterFactory(RuntimeTypeAdapterFactory.of(Message.class, "$type").registerSubtype(TimeMessage.class));
        Gson gson = builder.create();

        String json = gson.toJson(messages);

        Message[] outMessages = gson.fromJson(json, Message[].class);
        TimeMessage tm = (TimeMessage)outMessages[0];
    }
}
于 2013-06-10T15:06:04.997 回答