Jak określić format daty używany, gdy JAXB organizuje xsd: dateTime?

86

Gdy JAXB organizuje obiekt daty ( XMLGregorianCalendar) do elementu xsd: dateTime. Jak określić format wynikowego XML?

Na przykład: domyślny format danych używa milisekund <StartDate>2012-08-21T13:21:58.000Z</StartDate> , muszę pominąć milisekundy. <StartDate>2012-08-21T13:21:58Z</StartDate>

Jak mogę określić format wyjściowy / format daty, którego ma używać? Używam javax.xml.datatype.DatatypeFactorydo stworzenia XMLGregorianCalendarobiektu.

XMLGregorianCalendar xmlCal = datatypeFactory.newXMLGregorianCalendar(cal);
Young Fu
źródło

Odpowiedzi:

126

Możesz użyć, XmlAdapteraby dostosować sposób zapisywania typu daty w 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);
        }
    }

}

Następnie możesz użyć @XmlJavaTypeAdapteradnotacji, aby określić, że ma XmlAdapterbyć używany dla określonego pola / właściwości.

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

Korzystanie z pliku powiązań xjb:

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

wygeneruje powyższą adnotację.
(Przy końcu dodawania xjcnazw: xmlns:xjc="http://java.sun.com/xml/ns/jaxb/xjc")

bdoughan
źródło
2
Dzięki za tę odpowiedź! Czy można dodać adnotację przez xsd lub plik wiążący? Znalazłem tylko Twój często cytowany wpis na blogu o bindings.xml, ale myślę, że obejmuje to inne aspekty.
guerda,
9
Jak wspomniał @PeterRader, SimpleDateFormat nie jest bezpieczny dla wątków - jeśli dwa wątki miałyby wejść jednocześnie do Marshal lub Unmarshal, można uzyskać bardzo nieprzewidywalne wyniki. Byłoby to bardzo trudne do odtworzenia w normalnych testach, ale może się zdarzyć pod obciążeniem i byłoby niezwykle trudne do zdiagnozowania. Lepiej jest utworzyć nowy SimpleDateFormat z marszałkiem i unmarshalem (ale w razie potrzeby użyj statycznego ciągu formatu).
Colselaw
1
Zrobiłem to i prawie się udało. Jednak otrzymywałem Class has two properties of the same name "timeSeries"błąd - ten problem został rozwiązany poprzez umieszczenie adnotacji na poziomie pobierania, a nie na poziomie elementu członkowskiego. (Podziękowania dla @megathor ze stackoverflow.com/questions/6768544/… )
gordon613
1
@ gordon613 - Ten artykuł zawiera dodatkowe informacje o tym, gdzie umieścić adnotację: blog.bdoughan.com/2011/06/using-jaxbs-xmlaccessortype-to.html
bdoughan
3
Ponieważ blok krytyczny jest chroniony jako „zsynchronizowany”, nie ma żadnego problemu. Jeśli wykonanych zostanie wiele połączeń, wystąpi problem (z wydajnością).
Mike Argyriou
17

Używam SimpleDateFormat do tworzenia XMLGregorianCalendar, tak jak w tym przykładzie:

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));
}

Pierwsza metoda tworzy wystąpienie XMLGregorianCalendar, które jest formatowane przez XML Marshaller jako prawidłowy xsd: date, druga metoda daje prawidłowy xsd: dateTime.

Andrea Luciano
źródło
2

Bardzo łatwa droga do mnie. Formatowanie XMLGregorianCalendar do kierowania w Javie.

Po prostu tworzę dane w dobrym formacie. toStringZostanie wywołana wytwarzania dobry wynik.

public static final XMLGregorianCalendar getDate(Date d) {
    try {
        return DatatypeFactory.newInstance().newXMLGregorianCalendar(new SimpleDateFormat("yyyy-MM-dd").format(d));
    } catch (DatatypeConfigurationException e) {
        return null;
    }
}
Iván
źródło
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);
    }
}
Mikrofon
źródło
0

Stosowanie:

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);
    }
}
Mikrofon
źródło