web-dev-qa-db-fra.com

Conversion simple entre Java.util.Date et XMLGregorianCalendar

Je cherche une méthode simple de conversion entre Java.util.Date et javax.xml.datatype.XMLGregorianCalendar dans les deux sens.

Voici le code que j'utilise maintenant:

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

Y a-t-il quelque chose de plus simple, comme certains appels d'API que j'ai oubliés?

La conversion entre une date/heure XML standard et un objet date Java semble être une tâche assez routinière et je suis surpris de devoir écrire ce code.

Aucune suggestion?

NOTES: Mes classes JAXB sont générées automatiquement à partir d'un schéma. Le processus de construction de mon projet ne me permet pas d’apporter des modifications manuelles aux classes générées. Les éléments xs: dateTime sont générés par XJC en tant que XMLGregorianCalendar dans les classes JAXB. Le schéma est étendu et peaufiné périodiquement. Je suis donc autorisé à apporter des modifications limitées au fichier XSD du schéma.

MISE À JOUR SUR LA SOLUTION: La solution proposée par Blaise m'a permis de supprimer XMLGregorianCalendar du mélange et de traiter à la place des objets Java.util.Calendar. En ajoutant une clause de liaison JAXB en haut de mon fichier de schéma, XJC est capable de générer des mappages plus appropriés pour xs: dateTime dans mes classes JAXB. Voici quelques extraits qui montrent les modifications apportées à mon fichier XSD.

L'élément racine dans le fichier 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">

Bloc d'annotation de liaison JAXB, inséré immédiatement après l'élément racine dans 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>

Étant donné que le champ XML xs: dateTime stocke également le fuseau horaire, il serait peut-être préférable pour moi de travailler avec Calendar plutôt que Date de toute façon, car les objets Calendar ont une assez bonne API pour travailler avec les paramètres régionaux et les fuseaux horaires. Dans tous les cas, je suis beaucoup plus heureux de traiter avec des objets de calendrier au lieu de XMLGregorianCalendar. Plus besoin des méthodes de conversion que j'ai énumérées ci-dessus. Je ne me suis pas rendu jusqu'à Java.util.Date, mais assez près!

110
Jim Tough

Pourquoi ne pas utiliser un fichier de liaison externe pour indiquer à XJC de générer des champs Java.util.Date au lieu de XMLGregorianCalendar?

Voir aussi Comment mapper xs: date sur Java.util.Date? Blog

46
bdoughan

De XMLGregorianCalendar à Java.util.Date, vous pouvez simplement faire:

Java.util.Date dt = xmlGregorianCalendarInstance.toGregorianCalendar().getTime();  
81
Zé Carlos

De Java.util.Date à XMLGregorianCalendar, vous pouvez simplement faire:

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

Code édité après le premier commentaire de @ f-puras, car je fais une erreur.

J'ai dû faire quelques changements pour que cela fonctionne, car certaines choses semblent avoir changé entre-temps:

  • xjc se plaindrait que mon adaptateur n'allonge pas XmlAdapter
  • certaines importations bizarres et inutiles ont été dessinées (org.w3._2001.xmlschema)
  • les méthodes d'analyse ne doivent pas être statiques lors de l'extension de XmlAdapter, évidemment

Voici un exemple de travail, espérons que cela aide (j'utilise JodaTime mais dans ce cas, SimpleDate serait suffisant):

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

Dans le xsd, j'ai suivi les excellentes références données ci-dessus, j'ai donc inclus cette annotation 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>
5
Gregor

Moi aussi j'ai eu ce genre de mal de tête. Je m'en suis débarrassé en représentant simplement les champs temporels comme primitifs dans mon POJO. Maintenant, la génération de mon code client WS gère tout correctement et pas plus de merde XML-to-Java. Et bien sûr, traiter avec les millis côté Java est simple et sans douleur. KISS le principe est génial!

1
Alex

Vous pouvez utiliser cette personnalisation pour modifier le mappage par défaut en 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>
1
Dev

Personnalisation du calendrier et de la date pendant le marshaling

Étape 1: Préparez jaxb binding xml pour des propriétés personnalisées, dans ce cas, je me suis préparé pour la date et le calendrier

<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" />


Étape 2: Ajouter un fichier de liaison jaxb personnalisé à Apache ou à tout plugin associé à l'option xsd, comme mentionné ci-dessous

<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>

Étape 3: écrivez le code pour la classe 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);
    }
}

Étape 4: Sortie

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

   <fileDate>2014-09-25</fileDate> - Date class formatted
</xmlHeader>
0
Kanthishere