Простое преобразование между java.util.Date и XMLGregorianCalendar

110

Я ищу простой метод преобразования между java.util.Date и javax.xml.datatype.XMLGregorianCalendar в обоих направлениях.

Вот код, который я использую сейчас :

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

Есть ли что-нибудь попроще, например, вызов API, который я пропустил?

Преобразование между стандартной датой / временем XML и объектом даты Java кажется довольно рутинной задачей, и я удивлен, что мне вообще пришлось писать этот код.

Какие-либо предложения?

ПРИМЕЧАНИЯ. Мои классы JAXB автоматически генерируются из схемы. Процесс сборки в моем проекте не позволяет мне вручную вносить изменения в сгенерированные классы. Элементы xs: dateTime генерируются XJC как XMLGregorianCalendar в классах JAXB. Схема периодически расширяется и настраивается, поэтому мне разрешено вносить ограниченные изменения в файл XSD схемы.

ОБНОВЛЕНИЕ РЕШЕНИЯ . Решение, предложенное Блейзом, позволило мне исключить XMLGregorianCalendar и вместо этого работать с объектами java.util.Calendar. Добавляя предложение привязки JAXB в верхней части моего файла схемы, XJC может создавать более подходящие сопоставления для xs: dateTime в моих классах JAXB. Вот несколько фрагментов, которые показывают изменения в моем файле XSD.

Корневой элемент в файле 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">

Блок аннотации привязки JAXB, вставляемый сразу после корневого элемента в 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>

Поскольку в поле XML xs: dateTime также хранится часовой пояс, мне может быть лучше работать с Calendar вместо Date, поскольку объекты Calendar имеют довольно хороший API для работы с языковыми стандартами и часовыми поясами. В любом случае, мне гораздо приятнее иметь дело с объектами Calendar вместо XMLGregorianCalendar. Нет необходимости в методах преобразования, которые я перечислил выше. Я не дошел до java.util.Date, но достаточно близко!

Джим Таф
источник
Я ничего не знаю. Но у вас все в порядке - просто поместите его в utilпакет и используйте.
Божо
В некотором роде, но зачем вам вообще нужно иметь дело с объектами XMLGregorianCalendar? Они раздражают. Если они поступают из jaxb, можно использовать @XMLTypeAdapter для прямой привязки к java.util.Date. Конечно, если вы автоматически генерируете схему, изменение объектов может вызывать такое же раздражение при регенерации.
Affe
@Affe Я автоматически генерирую схему, поэтому я не могу вручную вносить изменения в сгенерированные классы JAXB
Джим Таф
Это то же самое, что и stackoverflow.com/questions/835889/… ?
Джейкоб Томо,
1
@Jacob - это не так. Он уже придумал, как это сделать, ему интересно, нет ли готового служебного класса.
Божо

Ответы:

46

Почему бы не использовать внешний файл привязки, чтобы XJC генерировал поля java.util.Date вместо XMLGregorianCalendar?

Также см. Как сопоставить xs: date с java.util.Date? Блог

bdoughan
источник
Я займусь этим. Спасибо.
Джим Таф
Нет проблем. JAXB может обрабатывать тип java.util.Date, вам просто нужно сгенерировать его в своей модели. Что может быть непросто.
bdoughan
Это сработало для меня. Подробнее о том, что я сделал, см. Выше правки к моему вопросу.
Джим Таф
Я добавил привязки jaxb, но прямо под схемой xs:, и я получаю следующую ошибку: com.sun.istack.SAXParseException2: компилятор не смог выполнить эту настройку globalBindings. Он прикреплен не в том месте или несовместим с другими привязками. в com.sun.tools.xjc.ErrorReceiver.error (ErrorReceiver.java:86) в ..
pri
@pritam - Вот еще один пример, который может помочь: blog.bdoughan.com/2011/08/xml-schema-to-java-generating.html . Возможно, стоит задать новый вопрос по проблеме, которую вы видите.
bdoughan
81

Из XMLGregorianCalendar в java.util.Date вы можете просто сделать:

java.util.Date dt = xmlGregorianCalendarInstance.toGregorianCalendar().getTime();  
Зе Карлос
источник
Спасибо ... Я искал способ конвертировать XMLGregorianCalendar во время в миллисекундах.
Andez
6

Из java.util.Date в XMLGregorianCalendar вы можете просто сделать:

import javax.xml.datatype.XMLGregorianCalendar;
import javax.xml.datatype.DatatypeFactory;
import java.util.GregorianCalendar;
......
GregorianCalendar gcalendar = new GregorianCalendar();
gcalendar.setTime(yourDate);
XMLGregorianCalendar xmlDate = DatatypeFactory.newInstance().newXMLGregorianCalendar(gcalendar);

Код отредактирован после первого комментария @ f-puras, потому что я делаю ошибку.

moralejaSinCuentoNiProverbio
источник
1
Не работает так, как вы написали: GregorianCalendar.setTime () ничего не вернет.
f_puras
5

Мне пришлось внести некоторые изменения, чтобы он заработал, поскольку некоторые вещи, похоже, за это время изменились:

  • xjc будет жаловаться, что мой адаптер не расширяет XmlAdapter
  • какой-то странный и ненужный импорт был нарисован в (org.w3._2001.xmlschema)
  • методы синтаксического анализа не должны быть статическими при расширении XmlAdapter, очевидно

Вот рабочий пример, надеюсь, это поможет (я использую JodaTime, но в этом случае SimpleDate будет достаточно):

import java.util.Date;
import javax.xml.bind.DatatypeConverter;
import javax.xml.bind.annotation.adapters.XmlAdapter;
import org.joda.time.DateTime;

public class DateAdapter extends XmlAdapter<Object, Object> {
    @Override
    public Object marshal(Object dt) throws Exception {
        return new DateTime((Date) dt).toString("YYYY-MM-dd");
    }

    @Override
        public Object unmarshal(Object s) throws Exception {
        return DatatypeConverter.parseDate((String) s).getTime();
    }
}

В xsd я следовал превосходным ссылкам, приведенным выше, поэтому я включил эту аннотацию xml:

<xsd:appinfo>
    <jaxb:schemaBindings>
        <jaxb:package name="at.mycomp.xml" />
    </jaxb:schemaBindings>
    <jaxb:globalBindings>
        <jaxb:javaType name="java.util.Date" xmlType="xsd:date"
              parseMethod="at.mycomp.xml.DateAdapter.unmarshal"
          printMethod="at.mycomp.xml.DateAdapter.marshal" />
    </jaxb:globalBindings>
</xsd:appinfo>
Грегор
источник
1
Я стал поклонником Joda Time с тех пор, как задал этот вопрос. Намного лучше, чем классы даты и времени Java SE. Отлично для обработки часовых поясов!
Джим Таф
1

У меня тоже была такая головная боль. Избавился от него, просто представив поля времени как примитивные длинные в моем POJO. Теперь генерация моего клиентского кода WS обрабатывает все правильно и больше никакой ерунды из XML в Java. И, конечно же, работа с millis на стороне Java проста и безболезненна. Принцип KISS потрясающий!

Alex
источник
1

Вы можете использовать эту настройку, чтобы изменить сопоставление по умолчанию на java.util.Date

<xsd:annotation>
<xsd:appinfo>
    <jaxb:globalBindings>
        <jaxb:javaType name="java.util.Date" xmlType="xsd:dateTime"
                 parseMethod="org.apache.cxf.xjc.runtime.DataTypeAdapter.parseDateTime"
                 printMethod="org.apache.cxf.xjc.runtime.DataTypeAdapter.printDateTime"/>
    </jaxb:globalBindings>
</xsd:appinfo>

Dev
источник
0

Настройка календаря и даты во время маршалинга

Шаг 1. Подготовьте привязку jaxb xml для настраиваемых свойств. В этом случае я подготовился к дате и календарю.

<jaxb:bindings version="2.1" xmlns:jaxb="http://java.sun.com/xml/ns/jaxb"
xmlns:xjc="http://java.sun.com/xml/ns/jaxb/xjc" 
xmlns:xs="http://www.w3.org/2001/XMLSchema">
<jaxb:globalBindings generateElementProperty="false">
<jaxb:serializable uid="1" />
<jaxb:javaType name="java.util.Date" xmlType="xs:date"
    parseMethod="org.apache.cxf.tools.common.DataTypeAdapter.parseDate"
    printMethod="com.stech.jaxb.util.CalendarTypeConverter.printDate" />
<jaxb:javaType name="java.util.Calendar" xmlType="xs:dateTime"
    parseMethod="javax.xml.bind.DatatypeConverter.parseDateTime"
    printMethod="com.stech.jaxb.util.CalendarTypeConverter.printCalendar" />


Setp 2: добавить пользовательский файл привязки jaxb в Apache или любые связанные плагины в опции xsd, как указано ниже.

<xsdOption>
  <xsd>${project.basedir}/src/main/resources/tutorial/xsd/yourxsdfile.xsd</xsd>
  <packagename>com.tutorial.xml.packagename</packagename>
  <bindingFile>${project.basedir}/src/main/resources/xsd/jaxbbindings.xml</bindingFile>
</xsdOption>

Setp 3: напишите код для класса CalendarConverter

package com.stech.jaxb.util;

import java.text.SimpleDateFormat;

/**
 * To convert the calendar to JaxB customer format.
 * 
 */

public final class CalendarTypeConverter {

    /**
     * Calendar to custom format print to XML.
     * 
     * @param val
     * @return
     */
    public static String printCalendar(java.util.Calendar val) {
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd'T'hh:mm:ss");
        return simpleDateFormat.format(val.getTime());
    }

    /**
     * Date to custom format print to XML.
     * 
     * @param val
     * @return
     */
    public static String printDate(java.util.Date val) {
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
        return simpleDateFormat.format(val);
    }
}

Setp 4: Выход

  <xmlHeader>
   <creationTime>2014-09-25T07:23:05</creationTime> Calendar class formatted

   <fileDate>2014-09-25</fileDate> - Date class formatted
</xmlHeader>
Кантишер
источник