Пользовательский класс сериализации JodaTime возвращает огромный объект

Мне нужно сериализовать Joda DateTime в JSON как одну строку. Я использую реализацию JsonSerializer, которая предполагает создание строки, но я получаю объект с множеством свойств:

"metrics":{"startTimestamp":{"year":2014,"dayOfMonth":11,"dayOfWeek":2,"era":1,"dayOfYear":315,"centuryOfEra":20,"yearOfCentury":14,"monthOfYear":11,"weekOfWeekyear":46,"millisOfSecond":505,"millisOfDay":36348505,"secondOfMinute":48,"secondOfDay":36348,"minuteOfHour":5,.......... etc

Но мне нужно что-то вроде dd.mm.yyyy HH:mm:ss.SSS. Как это сделать?

CustomDateSerializer

public class CustomDateSerializer extends JsonSerializer<DateTime> {
    private static DateTimeFormatter formatter = DateTimeFormat.forPattern("dd.MM.yyyy HH:mm:ss.SSS");

    @Override
    public void serialize(DateTime value, JsonGenerator gen, SerializerProvider arg2) throws IOException, JsonProcessingException {
        gen.writeString(formatter.print(value));
//        System.out.println(formatter.print(value));
    }
    @Override
    public Class<DateTime> handledType() {
        return DateTime.class;
    }
}

Показатели

    import app.service.CustomDateSerializer;
import org.codehaus.jackson.annotate.JsonIgnore;
import org.codehaus.jackson.map.annotate.JsonSerialize;
import org.joda.time.DateTime;
import org.joda.time.Period;
import org.joda.time.format.PeriodFormatter;
import org.joda.time.format.PeriodFormatterBuilder;

public class Metrics {
//    @JsonSerialize(using = CustomDateSerializer.class)
    DateTime startTimestamp;
//    @JsonSerialize(using = CustomDateSerializer.class)
    DateTime endTimestamp;
    Period period;
    @JsonIgnore
    PeriodFormatter periodFormatter;

    public Metrics() {

    }
    public DateTime getStartTimestamp() {
        return startTimestamp;
    }

    public void setStartTimestamp(DateTime startTimestamp) {
        this.startTimestamp = startTimestamp;
    }

    public DateTime getEndTimestamp() {
        return endTimestamp;
    }

    public void setEndTimestamp(DateTime endTimestamp) {
        this.endTimestamp = endTimestamp;
    }

    public Period getPeriod() {
        return period;
    }
    public void setPeriod() {
        this.period = new Period(this.startTimestamp, this.endTimestamp);
    }
    public void setPeriod(Period period) {
        this.period = period;
    }

    public PeriodFormatter getPeriodFormatter() {
        return periodFormatter;
    }

    public void setPeriodFormatter(PeriodFormatter periodFormatter) {
        this.periodFormatter = periodFormatter;
    }
   @Override
   public String toString(){
       this.periodFormatter = new PeriodFormatterBuilder()
                .printZeroAlways()
                .minimumPrintedDigits(2)
                .appendHours().appendSeparator(":")
                .appendMinutes().appendSeparator(":")
                .appendSeconds().appendSeparator(".")
                .appendMillis3Digit()
                .toFormatter();
       return "Started: " + this.startTimestamp.toString() + "\\n" + "Ended:   " + this.endTimestamp.toString() + "\\n" + "Response took: " + periodFormatter.print(period)+ "";
   }
}

ОБНОВЛЕНИЕ:

System.out.println(formatter.print(value)) в CustomDateSerializer на самом деле печатает правильную строку, но по какой-то причине он не смог передать ее сериализатору, я полагаю ...

ОБНОВЛЕНИЕ2

Контроллер

public @ResponseBody XmlResponse getGUID( @RequestParam(/*...*/) String environmentParam) {
//.... XmlResponse xmlResponse = ..........
return xmlResponse;
}

Класс XmlResponse

import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.datatype.joda.JodaModule;
import org.codehaus.jackson.JsonProcessingException;
import org.codehaus.jackson.annotate.JsonIgnore;
import org.codehaus.jackson.annotate.JsonProperty;
import org.codehaus.jackson.map.ObjectMapper;

import java.io.IOException;

public class XmlResponse {
    String xmlResponseBody;
    @JsonIgnore
    Metrics metrics;
    Boolean error;

    public XmlResponse() {
    }

    public XmlResponse(String xmlResponseBody, Metrics metrics, Boolean error) {
        this.xmlResponseBody = xmlResponseBody;
        this.metrics = metrics;
        this.error = error;
    }

    public String getXmlResponseBody() {
        return xmlResponseBody;
    }

    public void setXmlResponseBody(String xmlResponseBody) {
        this.xmlResponseBody = xmlResponseBody;
    }
    @JsonProperty("metrics")
    public Metrics getMetrics() {
        return metrics;
    }

    public void setMetrics(Metrics metrics) {
        this.metrics = metrics;
    }

    public Boolean getError() {
        return error;
    }

    public void setError(Boolean error) {
        this.error = error;
    }

    @Override
    public String toString(){
        String out = "";
        ObjectMapper mapper = new ObjectMapper();
        try {
//            mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
//            mapper.registerModule(new JodaModule());
             out = mapper.writeValueAsString(this);
        } catch (JsonProcessingException e) {
            e.printStackTrace();
            return "{}";
        } catch (IOException e) {
            e.printStackTrace();
        }
        return out;
    }
}

ОБНОВЛЕНИЕ3

раскомментирование @JsonSerialize(using = CustomDateSerializer.class) не меняет результат


person J.Olufsen    schedule 11.11.2014    source источник
comment
Кажется, ваш код дает мне ожидаемый результат с ObjectMapper#writeValue. Как именно вы выполняете сериализацию?   -  person izstas    schedule 11.11.2014


Ответы (1)


Ваш код представляет собой смесь Jackson 1.x (org.codehaus.jackson) и Jackson 2.x (com.fasterxml.jackson). Хотя их API-интерфейсы очень похожи, аннотации Jackson 1.x не влияют на Jackson 2.x и наоборот (если вы не используете jackson-legacy-introspector).

Убедитесь, что вы везде используете одну и ту же версию Jackson, это поможет решить вашу проблему.
Вы можете получить дополнительную информацию здесь: Обновление Jackson 1.9 до 2.0.

person izstas    schedule 11.11.2014