Come si specifica il formato della data utilizzato quando JAXB esegue il marshalling di xsd: dateTime?


86

Quando JAXB effettua il marshalling di un oggetto date ( XMLGregorianCalendar) in un elemento xsd: dateTime. Come puoi specificare il formato dell'XML risultante?

Ad esempio: il formato dati predefinito utilizza i millisecondi di cui <StartDate>2012-08-21T13:21:58.000Z</StartDate> ho bisogno per omettere i millisecondi. <StartDate>2012-08-21T13:21:58Z</StartDate>

Come posso specificare il modulo di output / formato della data che voglio che utilizzi? Sto usando javax.xml.datatype.DatatypeFactoryper creare l' XMLGregorianCalendaroggetto.

XMLGregorianCalendar xmlCal = datatypeFactory.newXMLGregorianCalendar(cal);

Risposte:


126

È possibile utilizzare un XmlAdapterper personalizzare il modo in cui un tipo di data viene scritto in XML.

package com.example;

import java.text.SimpleDateFormat;
import java.util.Date;

import javax.xml.bind.annotation.adapters.XmlAdapter;

public class DateAdapter extends XmlAdapter<String, Date> {

    private final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    @Override
    public String marshal(Date v) throws Exception {
        synchronized (dateFormat) {
            return dateFormat.format(v);
        }
    }

    @Override
    public Date unmarshal(String v) throws Exception {
        synchronized (dateFormat) {
            return dateFormat.parse(v);
        }
    }

}

Quindi si utilizza l' @XmlJavaTypeAdapterannotazione per specificare che XmlAdapterdeve essere utilizzato per un campo / proprietà specifico.

@XmlElement(name = "timestamp", required = true) 
@XmlJavaTypeAdapter(DateAdapter.class)
protected Date timestamp; 

Utilizzando un file di associazione xjb:

<xjc:javaType name="java.util.Date" xmlType="xs:dateTime"
        adapter="com.example.DateAdapter"/>

produrrà l'annotazione di cui sopra.
(Con l'aggiunta eventualmente il xjcnamespace: xmlns:xjc="http://java.sun.com/xml/ns/jaxb/xjc")


2
Grazie per questa risposta! È possibile aggiungere l'annotazione tramite xsd o un file di binding? Ho trovato solo il tuo post di blog citato di frequente su bindings.xml, ma questo copre altri aspetti, credo.
guerda

9
Come menzionato da @PeterRader, SimpleDateFormat non è thread-safe: se due thread dovessero entrare in marshalling o unmarshal simultaneamente, potresti ottenere risultati molto imprevedibili. Questo sarebbe molto difficile da riprodurre durante i normali test, ma sotto carico potrebbe accadere e sarebbe estremamente difficile da diagnosticare. È preferibile creare un nuovo SimpleDateFormat con marshal e unmarshal (ma utilizzare una stringa di formato statica se necessario).
Colselaw

1
L'ho fatto e ha quasi funzionato. Tuttavia stavo ricevendo un Class has two properties of the same name "timeSeries"errore: questo è stato risolto inserendo l'annotazione a livello di getter e non a livello di membro. (Grazie a @megathor da stackoverflow.com/questions/6768544/... )
gordon613

1
@ gordon613 - Questo articolo fornirà alcune informazioni aggiuntive su dove inserire l'annotazione: blog.bdoughan.com/2011/06/using-jaxbs-xmlaccessortype-to.html
bdoughan

3
Poiché il blocco critico è protetto con "sincronizzato" non c'è alcun problema. Ci saranno problemi (di prestazioni) se vengono effettuate più chiamate.
Mike Argyriou

17

Uso un SimpleDateFormat per creare XMLGregorianCalendar, come in questo esempio:

public static XMLGregorianCalendar getXmlDate(Date date) throws DatatypeConfigurationException {
    return DatatypeFactory.newInstance().newXMLGregorianCalendar(new SimpleDateFormat("yyyy-MM-dd").format(date));
}

public static XMLGregorianCalendar getXmlDateTime(Date date) throws DatatypeConfigurationException {
    return DatatypeFactory.newInstance().newXMLGregorianCalendar(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss").format(date));
}

Il primo metodo crea un'istanza di XMLGregorianCalendar formattata dal marshaller XML come xsd: date valido, il secondo metodo restituisce un xsd: dateTime valido.


2

Modo molto semplice per me. Formattazione XMLGregorianCalendar per il marshalling in java.

Creo solo i miei dati nel buon formato. La toStringsi chiamerà producendo il buon risultato.

public static final XMLGregorianCalendar getDate(Date d) {
    try {
        return DatatypeFactory.newInstance().newXMLGregorianCalendar(new SimpleDateFormat("yyyy-MM-dd").format(d));
    } catch (DatatypeConfigurationException e) {
        return null;
    }
}

1

https://www.baeldung.com/jaxb

public class DateAdapter extends XmlAdapter<String, Date> {

    private static final ThreadLocal<DateFormat> dateFormat 
      = new ThreadLocal<DateFormat>() {

        @Override
        protected DateFormat initialValue() {
            return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        }
    }

    @Override
    public Date unmarshal(String v) throws Exception {
        return dateFormat.get().parse(v);
    }

    @Override
    public String marshal(Date v) throws Exception {
        return dateFormat.get().format(v);
    }
}

0

Utilizzo:

import com.company.LocalDateAdapter.yyyyMMdd;
...

@XmlElement(name = "PROC-DATE")
@XmlJavaTypeAdapter(yyyyMMdd.class)
private LocalDate processingDate;

LocalDateAdapter

import javax.xml.bind.annotation.adapters.XmlAdapter;
import org.joda.time.LocalDate;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;

public class LocalDateAdapter extends XmlAdapter<String, LocalDate> {

    public static final class yyyyMMdd extends LocalDateAdapter {
        public yyyyMMdd() {
            super("yyyyMMdd");
        }
    }

    public static final class yyyy_MM_dd extends LocalDateAdapter {
        public yyyy_MM_dd() {
            super("yyyy-MM-dd");
        }
    }

    private final DateTimeFormatter formatter;

    public LocalDateAdapter(String pattern) {
        formatter = DateTimeFormat.forPattern(pattern);
    }

    @Override
    public String marshal(LocalDate date) throws Exception {
        return formatter.print(date);
    }

    @Override
    public LocalDate unmarshal(String date) throws Exception {
        return formatter.parseLocalDate(date);
    }
}
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.