Как указать формат даты, используемый при маршале JAXB xsd: dateTime?

86

Когда JAXB маршалирует объект даты ( XMLGregorianCalendar) в элемент xsd: dateTime. Как указать формат результирующего XML?

Например: в формате данных по умолчанию используются миллисекунды, <StartDate>2012-08-21T13:21:58.000Z</StartDate> мне нужно опустить миллисекунды. <StartDate>2012-08-21T13:21:58Z</StartDate>

Как я могу указать формат вывода / даты, который я хочу использовать? Я использую javax.xml.datatype.DatatypeFactoryдля создания XMLGregorianCalendarобъекта.

XMLGregorianCalendar xmlCal = datatypeFactory.newXMLGregorianCalendar(cal);
Молодой фу
источник

Ответы:

126

Вы можете использовать, XmlAdapterчтобы настроить, как тип даты записывается в 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);
        }
    }

}

Затем вы используете @XmlJavaTypeAdapterаннотацию, чтобы указать, что XmlAdapterследует использовать для определенного поля / свойства.

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

Используя файл привязки xjb:

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

создаст вышеупомянутую аннотацию.
(К в конце концов добавив xjcпространство имен: xmlns:xjc="http://java.sun.com/xml/ns/jaxb/xjc")

Bdoughan
источник
2
Спасибо за ответ! Можно ли добавить аннотацию через xsd или файл привязки? Я нашел только вашу часто цитируемую запись в блоге о bindings.xml, но я думаю, что она охватывает другие аспекты.
guerda
9
Как упоминалось в @PeterRader, SimpleDateFormat не является потокобезопасным - если два потока одновременно будут вводить маршалинг или демаршалинг, вы можете получить очень непредсказуемые результаты. Это было бы очень трудно воспроизвести при обычном тестировании, но под нагрузкой могло бы случиться и было бы чрезвычайно сложно диагностировать. Лучше создать новый SimpleDateFormat с маршалингом и демаршалингом (но при необходимости используйте строку статического формата).
Colselaw
1
Я сделал это, и это почти сработало. Однако я получал Class has two properties of the same name "timeSeries"ошибку - это было решено путем размещения аннотации в получателе, а не на уровне члена. (Спасибо @megathor из stackoverflow.com/questions/6768544/… )
gordon613
1
@ gordon613 - Эта статья предоставит дополнительную информацию о том, где разместить аннотацию: blog.bdoughan.com/2011/06/using-jaxbs-xmlaccessortype-to.html
bdoughan
3
Поскольку критический блок защищен "синхронизированным", проблем нет. При выполнении нескольких вызовов возникнут проблемы (с производительностью).
Майк Аргириу,
17

Я использую SimpleDateFormat для создания XMLGregorianCalendar, например, в этом примере:

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

Первый метод создает экземпляр XMLGregorianCalendar, который форматируется XML-маршаллером как допустимый xsd: date, второй метод дает допустимый xsd: dateTime.

Андреа Лучано
источник
2

Для меня очень простой способ. Форматирование XMLGregorianCalendar для маршалинга в java.

Я просто создаю свои данные в хорошем формате. Это toStringбудет называться производящим хороший результат.

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

Применение:

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);
    }
}
Майк
источник