Як ви вказуєте формат дати, який використовується, коли маршали 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

Ви можете використовувати a, 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")


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
Оскільки критичний блок захищений синхронізацією, проблем не виникає. Буде проблема (продуктивності), якщо буде здійснено кілька дзвінків.
Mike Argyriou

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);
    }
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.