Полиморфная сериализация Джексона порождает неверное имя класса

Когда я использую полиморфную сериализацию Джексона, она генерирует объект JSON с неверным полностью определенным именем класса.

Код ниже сериализует XMLGregorianCalendar. Выход:

["java.util.GregorianCalendar",-3600000]

Я ожидал следующего:

["javax.xml.datatype.XMLGregorianCalendar",-3600000]

Почему он выводит java.util.GregorianCalendar?

Или, что более важно: как мне это исправить?

Пример кода:

import com.fasterxml.jackson.databind.ObjectMapper;

import javax.xml.datatype.DatatypeConfigurationException;
import javax.xml.datatype.DatatypeFactory;
import javax.xml.datatype.XMLGregorianCalendar;
import java.io.ByteArrayOutputStream;

public class JacksonGregorianProblem {

    public static void main(String[] args) throws java.io.IOException, DatatypeConfigurationException {

        XMLGregorianCalendar xmlGregorianCalendar = DatatypeFactory.newInstance().newXMLGregorianCalendar();
        ObjectMapper mapper = new ObjectMapper();
        mapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);

        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        mapper.writeValue(byteArrayOutputStream, xmlGregorianCalendar);

        System.out.println(byteArrayOutputStream);
    }
}

2 ответа

Решение

Чтобы получить ожидаемое поведение, я реализовал пользовательский сериализатор XMLGregorianCalendar. Этот класс заботится о сериализации XLMGregorianCalendar, и теперь вывод - именно то, что я ожидаю.:-)

class XMLGregorianCalendarSerializer extends StdSerializer<XMLGregorianCalendar> {

    public XMLGregorianCalendarSerializer() {
        this(null);
    }

    public XMLGregorianCalendarSerializer(Class<XMLGregorianCalendar> t) {
        super(t);
    }

    @Override
    public void serialize(XMLGregorianCalendar value, JsonGenerator gen, SerializerProvider provider)
        throws IOException
    {
        gen.writeNumber(value.toGregorianCalendar().getTimeInMillis());
    }

    @Override
    public void serializeWithType(XMLGregorianCalendar value, JsonGenerator gen, SerializerProvider provider,
        TypeSerializer typeSerializer) throws IOException
    {
        gen.writeStartArray();
        gen.writeString("javax.xml.datatype.XMLGregorianCalendar");
        serialize(value, gen, provider); // call your customized serialize method
        gen.writeEndArray();
    }
}

Вы можете добавить этот сериализатор к объекту сопоставления с кодом ниже. Это можно вставить в пример кода в вопросе.

    SimpleModule module = new SimpleModule();
    module.addSerializer(XMLGregorianCalendar.class, new XMLGregorianCalendarSerializer());
    mapper.registerModule(module);

XML-григорианский календарь предполагает, что только сериализация и десериализация обрабатываются. Поэтому я предпочитаю использовать следующий сериализатор и десериализатор:

      private class XMLCalendarDeserializer extends StdDeserializer<XMLGregorianCalendar> {

        private DatatypeFactory factory = DatatypeFactory.newDefaultInstance();
        
        public XMLCalendarDeserializer() {
            super(XMLGregorianCalendar.class);
        }

        
        
        @Override
        public XMLGregorianCalendar deserialize(JsonParser parser, DeserializationContext ctx) throws IOException, JsonProcessingException {
            if (parser.hasToken(JsonToken.VALUE_STRING)) {
                return factory.newXMLGregorianCalendar(parser.getText());
            } else {
                throw new JsonParseException(parser, "not string token");
            }
        }
        
    }
    
    private class XMLCalendarSerializer extends StdSerializer<XMLGregorianCalendar> {

        public XMLCalendarSerializer() {
            super(XMLGregorianCalendar.class);
        }
        
        @Override
        public void serialize(XMLGregorianCalendar val, JsonGenerator gen, SerializerProvider ser) throws IOException {
            gen.writeString(val.toXMLFormat());
        }
        
    }

`

Другие вопросы по тегам