Szukam prostej metody konwersji między java.util.Date i javax.xml.datatype.XMLGregorianCalendar w obu kierunkach.
Oto kod, którego teraz używam :
import java.util.GregorianCalendar;
import javax.xml.datatype.DatatypeConfigurationException;
import javax.xml.datatype.DatatypeFactory;
import javax.xml.datatype.XMLGregorianCalendar;
/**
* Utility class for converting between XMLGregorianCalendar and java.util.Date
*/
public class XMLGregorianCalendarConverter {
/**
* Needed to create XMLGregorianCalendar instances
*/
private static DatatypeFactory df = null;
static {
try {
df = DatatypeFactory.newInstance();
} catch (DatatypeConfigurationException dce) {
throw new IllegalStateException(
"Exception while obtaining DatatypeFactory instance", dce);
}
}
/**
* Converts a java.util.Date into an instance of XMLGregorianCalendar
*
* @param date Instance of java.util.Date or a null reference
* @return XMLGregorianCalendar instance whose value is based upon the
* value in the date parameter. If the date parameter is null then
* this method will simply return null.
*/
public static XMLGregorianCalendar asXMLGregorianCalendar(java.util.Date date) {
if (date == null) {
return null;
} else {
GregorianCalendar gc = new GregorianCalendar();
gc.setTimeInMillis(date.getTime());
return df.newXMLGregorianCalendar(gc);
}
}
/**
* Converts an XMLGregorianCalendar to an instance of java.util.Date
*
* @param xgc Instance of XMLGregorianCalendar or a null reference
* @return java.util.Date instance whose value is based upon the
* value in the xgc parameter. If the xgc parameter is null then
* this method will simply return null.
*/
public static java.util.Date asDate(XMLGregorianCalendar xgc) {
if (xgc == null) {
return null;
} else {
return xgc.toGregorianCalendar().getTime();
}
}
}
Czy jest coś prostszego, na przykład wywołanie interfejsu API, które przeoczyłem?
Konwersja między standardową datą / godziną XML a obiektem daty Java wydaje się dość rutynowym zadaniem i jestem zaskoczony, że w ogóle muszę napisać ten kod.
Jakieś sugestie?
UWAGI: Moje klasy JAXB są generowane automatycznie ze schematu. Proces kompilacji w moim projekcie nie pozwala mi na ręczne wprowadzanie zmian w wygenerowanych klasach. Elementy xs: dateTime są generowane przez XJC jako XMLGregorianCalendar w klasach JAXB. Schemat jest okresowo rozszerzany i modyfikowany, więc mogę wprowadzać ograniczone zmiany w pliku XSD schematu.
AKTUALIZACJA ROZWIĄZANIA: Rozwiązanie zaproponowane przez Blaise'a pozwoliło mi wyeliminować XMLGregorianCalendar i zająć się obiektami java.util.Calendar. Dodając klauzulę powiązania JAXB na górze mojego pliku schematu, XJC jest w stanie wygenerować bardziej odpowiednie mapowania dla xs: dateTime w moich klasach JAXB. Oto kilka fragmentów, które pokazują modyfikacje w moim pliku XSD.
Element główny w pliku XSD:
<xs:schema xmlns:mydata="http://my.example.com/mydata" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:jaxb="http://java.sun.com/xml/ns/jaxb" targetNamespace="http://my.example.com/mydata" elementFormDefault="unqualified" attributeFormDefault="unqualified" version="0.2" xml:lang="en" jaxb:version="2.0">
Blok adnotacji wiązania JAXB, wstawiany bezpośrednio po elemencie głównym w XSD:
<xs:annotation>
<xs:appinfo>
<jaxb:globalBindings>
<jaxb:javaType name="java.util.Calendar" xmlType="xs:dateTime" parseMethod="javax.xml.bind.DatatypeConverter.parseDateTime" printMethod="javax.xml.bind.DatatypeConverter.printDateTime" />
</jaxb:globalBindings>
</xs:appinfo>
</xs:annotation>
Ponieważ pole XML xs: dateTime również przechowuje strefę czasową, może lepiej byłoby dla mnie pracować z Kalendarzem zamiast z datą, ponieważ obiekty kalendarza mają całkiem niezłe API do pracy z ustawieniami narodowymi i strefami czasowymi. W każdym razie jestem o wiele szczęśliwszy, mogąc zajmować się obiektami Calendar zamiast XMLGregorianCalendar. Nie ma już potrzeby stosowania metod konwersji, które wymieniłem powyżej. Nie dotarłem do java.util.Date, ale wystarczająco blisko!
util
opakowaniu i użyj.