Come serializzare Joda DateTime con il processore Jackson JSON?


118

Come faccio a convincere Jackson a serializzare il mio oggetto Joda DateTime secondo uno schema semplice (come "dd-MM-yyyy")?

Ho provato:

@JsonSerialize(using=DateTimeSerializer.class)
private final DateTime date;

Ho anche provato:

ObjectMapper mapper = new ObjectMapper()
    .getSerializationConfig()
    .setDateFormat(df);

Grazie!


Entrambi i precedenti dovrebbero effettivamente funzionare (@JsonSerialize dovrebbe implicare che il campo deve essere serializzato; e il formato della data dovrebbe idealmente applicarsi anche a Joda), quindi potresti voler presentare un bug Jira su jira.codehaus.org/browse/JACKSON .
StaxMan

Mi rendo conto che questa domanda è di un po 'di tempo fa, ma per riferimento futuro, objectMapper.getSerializationConfig (). SetDateFormat (df) è ora deprecato. objectMapper.setDateFormat (df) è ora suggerito.
Patrick

Risposte:


146

Questo è diventato molto semplice con Jackson 2.0 e il modulo Joda.

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());

Dipendenza da Maven:

<dependency>
  <groupId>com.fasterxml.jackson.datatype</groupId>
  <artifactId>jackson-datatype-joda</artifactId>
  <version>2.1.1</version>
</dependency>  

Codice e documentazione: https://github.com/FasterXML/jackson-datatype-joda

Binari: http://repo1.maven.org/maven2/com/fasterxml/jackson/datatype/jackson-datatype-joda/


13
Potresti approfondire questo argomento per favore? Dove va questo codice e come viene utilizzata l'istanza di ObjectMapper?
Paul

Dovresti porre domande specifiche sull'uso dell'api di Jackson e di Maven. La posizione del codice dipende dall'utilizzo o meno di framework / dal modo in cui si avvia l'applicazione.
Kimble

4
quando aggiungo che ottengo l'errore di compilazione "tipi incompatibili: JodaModule non può essere convertito in Modulo" - il metodo si aspetta un org.codehaus.jackson.map.Module ma JodaModule non lo ha nella sua gerarchia, quindi come potrebbe funzionare?
Martin Charlesworth,

4
Promettente ma attualmente quasi inutile in quanto non possiamo specificare il formato della data da analizzare :(
Vincent Mimoun-Prat

10
È necessario disabilitare la serializzazione come timestamp objectMapper.disable (SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
MarekM

74

Nell'oggetto che stai mappando:

@JsonSerialize(using = CustomDateSerializer.class)
public DateTime getDate() { ... }

In CustomDateSerializer:

public class CustomDateSerializer extends JsonSerializer<DateTime> {

    private static DateTimeFormatter formatter = 
        DateTimeFormat.forPattern("dd-MM-yyyy");

    @Override
    public void serialize(DateTime value, JsonGenerator gen, 
                          SerializerProvider arg2)
        throws IOException, JsonProcessingException {

        gen.writeString(formatter.print(value));
    }
}

4
Come registrare CustomDateSerializer in spring mvc 3?
digz6666

1
Non so di Spring MVC3, ma puoi aggiungere un serializzatore personalizzato all'oggetto mappatore Jackson. Vedere questa voce in Jackson È necessario creare un modulo semplice per quel SimpleModule isoDateTimeModule = new SimpleModule ("ISODateTimeModule", new Version (1, 0, 0, null)); isoDateTimeModule.addSerializer (new JsonISODateTimeFormatSerializer ()); mapper.registerModule (isoDateTimeModule);
Guillaume Belrose,

1
@ digz6666 vedi la mia risposta qui per Spring MVC 3: stackoverflow.com/questions/7854030/…
Aram Kocharyan

Nota: assicurati di utilizzare le classi precedenti da org.codehaus.jacksonpiuttosto che com.fasterxml.jackson.core. L'utilizzo del secondo pacchetto non ha funzionato per me. In effetti, la mia app Jersey non rispettava nemmeno l' @JsonSerializedannotazione.
Kevin Meredith

La risposta ha funzionato per me. Ma richiederebbe questa annotazione su tutti i campi. C'è qualche configurazione globale per questo che funziona con Jackson?
Taher

26

Come ha detto @Kimble, con Jackson 2, utilizzare la formattazione predefinita è molto semplice; registrati semplicemente JodaModulesul tuo ObjectMapper.

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());

Per la serializzazione / deserializzazione personalizzata di DateTime, è necessario implementare il proprio StdScalarSerializere StdScalarDeserializer; è piuttosto contorto, ma comunque.

Ad esempio, ecco un file DateTime serializzatore che utilizza il ISODateFormatcon il fuso orario UTC:

public class DateTimeSerializer extends StdScalarSerializer<DateTime> {

    public DateTimeSerializer() {
        super(DateTime.class);
    }

    @Override
    public void serialize(DateTime dateTime,
                          JsonGenerator jsonGenerator,
                          SerializerProvider provider) throws IOException, JsonGenerationException {
        String dateTimeAsString = ISODateTimeFormat.withZoneUTC().print(dateTime);
        jsonGenerator.writeString(dateTimeAsString);
    }
}

E il corrispondente de-serializzatore:

public class DateTimeDesrializer extends StdScalarDeserializer<DateTime> {

    public DateTimeDesrializer() {
        super(DateTime.class);
    }

    @Override
    public DateTime deserialize(JsonParser jsonParser,
                                DeserializationContext deserializationContext) throws IOException, JsonProcessingException {
        try {
            JsonToken currentToken = jsonParser.getCurrentToken();
            if (currentToken == JsonToken.VALUE_STRING) {
                String dateTimeAsString = jsonParser.getText().trim();
                return ISODateTimeFormat.withZoneUTC().parseDateTime(dateTimeAsString);
            }
        } finally {
            throw deserializationContext.mappingException(getValueClass());
        }
    }

Quindi collegali insieme a un modulo:

public class DateTimeModule extends SimpleModule {

    public DateTimeModule() {
        super();
        addSerializer(DateTime.class, new DateTimeSerializer());
        addDeserializer(DateTime.class, new DateTimeDeserializer());
    }
}

Quindi registra il modulo sul tuo ObjectMapper:

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new DateTimeModule());

l'oggetto DatatIme deserilazed è nel fuso orario UTC oppure nel fuso orario del browser per favore aiutatemi
user3354457

Questo registra il mappatore (e i moduli) a livello globale?
raffian

16

La soluzione facile

Ho riscontrato un problema simile e la mia soluzione è molto chiara rispetto a quella precedente.

Ho semplicemente usato il modello in @JsonFormat annotazione

Fondamentalmente la mia classe ha un DateTimecampo, quindi metto un'annotazione attorno al getter:

@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
public DateTime getDate() {
    return date;
}

Serializzo la classe con ObjectMapper

    ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new JodaModule());
    mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
    ObjectWriter ow = mapper.writer();
    try {
        String logStr = ow.writeValueAsString(log);
        outLogger.info(logStr);
    } catch (IOException e) {
        logger.warn("JSON mapping exception", e);
    }

Usiamo Jackson 2.5.4


15

https://stackoverflow.com/a/10835114/1113510

Sebbene tu possa inserire un'annotazione per ogni campo data, è meglio fare una configurazione globale per il tuo mappatore oggetti. Se usi jackson puoi configurare la tua molla come segue:

<bean id="jacksonObjectMapper" class="com.company.CustomObjectMapper" />

<bean id="jacksonSerializationConfig" class="org.codehaus.jackson.map.SerializationConfig"
    factory-bean="jacksonObjectMapper" factory-method="getSerializationConfig" >
</bean>

Per CustomObjectMapper:

public class CustomObjectMapper extends ObjectMapper {

    public CustomObjectMapper() {
        super();
        configure(Feature.WRITE_DATES_AS_TIMESTAMPS, false);
        setDateFormat(new SimpleDateFormat("EEE MMM dd yyyy HH:mm:ss 'GMT'ZZZ (z)"));
    }
}

Naturalmente, SimpleDateFormat può utilizzare qualsiasi formato di cui hai bisogno.


4
setDateFormat(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssz"));ha fatto il trucco per me, grazie
Konsumierer

8

Nel frattempo Jackson registra automaticamente il modulo Joda quando JodaModule è in classpath. Ho appena aggiunto jackson-datatype-joda a Maven e ha funzionato immediatamente.

<dependency>
  <groupId>com.fasterxml.jackson.datatype</groupId>
  <artifactId>jackson-datatype-joda</artifactId>
  <version>2.8.7</version>
</dependency>

Uscita JSON:

{"created" : "2017-03-28T05:59:27.258Z"}

6

Per quelli con Spring Boot devi aggiungere il modulo al tuo contesto e verrà aggiunto alla tua configurazione in questo modo.

@Bean
public Module jodaTimeModule() {
    return new JodaModule();
}

E se vuoi usare il nuovo modulo java8 time jsr-310.

@Bean
public Module jodaTimeModule() {
    return new JavaTimeModule();
}

3

Sembra che per Jackson 1.9.12 non ci sia tale possibilità per impostazione predefinita, a causa di:

public final static class DateTimeSerializer
    extends JodaSerializer<DateTime>
{
    public DateTimeSerializer() { super(DateTime.class); }

    @Override
    public void serialize(DateTime value, JsonGenerator jgen, SerializerProvider provider)
        throws IOException, JsonGenerationException
    {
        if (provider.isEnabled(SerializationConfig.Feature.WRITE_DATES_AS_TIMESTAMPS)) {
            jgen.writeNumber(value.getMillis());
        } else {
            jgen.writeString(value.toString());
        }
    }

    @Override
    public JsonNode getSchema(SerializerProvider provider, java.lang.reflect.Type typeHint)
    {
        return createSchemaNode(provider.isEnabled(SerializationConfig.Feature.WRITE_DATES_AS_TIMESTAMPS)
                ? "number" : "string", true);
    }
}

Questa classe serializza i dati utilizzando il metodo toString () di Joda DateTime.

L'approccio proposto da Rusty Kuntz funziona perfettamente per il mio caso.


È ancora possibile registrare il serializzatore e il deserializzatore personalizzati e questi avranno la precedenza sull'implementazione predefinita.
StaxMan

2

Sto usando Java 8 e questo ha funzionato per me.

Aggiungi la dipendenza da pom.xml

<dependency>
    <groupId>com.fasterxml.jackson.datatype</groupId>
    <artifactId>jackson-datatype-jsr310</artifactId>
    <version>2.4.0</version>
</dependency>

e aggiungi JodaModule sul tuo ObjectMapper

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.