簡體   English   中英

java.util.Date和XMLGregorianCalendar之間的簡單轉換

[英]Simple conversion between java.util.Date and XMLGregorianCalendar

我正在尋找一種在兩個方向上在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在JAXB類中生成為XMLGregorianCalendar。 模式會定期擴展和調整,因此我可以對模式XSD文件進行有限的更改。

解決方案更新: Blaise提出的解決方案允許我從混合中獲取XMLGregorianCalendar並處理java.util.Calendar對象。 通過在我的模式文件的頂部添加JAXB綁定子句,XJC能夠在我的JAXB類中為xs:dateTime生成更合適的映射。 以下是一些顯示我的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,但足夠接近!

從XMLGregorianCalendar到java.util.Date,你可以簡單地做到:

java.util.Date dt = xmlGregorianCalendarInstance.toGregorianCalendar().getTime();  

為什么不使用外部綁定文件來告訴XJC生成java.util.Date字段而不是XMLGregorianCalendar?

另請參閱如何將xs:date映射到java.util.Date? 博客

從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的第一個評論之后編輯的代碼,因為我做錯了。

我不得不做一些改變才能使它工作,因為有些事情似乎在此期間發生了變化:

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

您可以使用此自定義將默認映射更改為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>

我也有這種頭痛。 通過簡單地在我的POJO中將時間字段表示為原始長度來擺脫它。 現在,我的WS客戶端代碼的生成正確地處理了所有內容,而不再是XML到Java的廢話。 當然,在Java方面處理毫秒是簡單而輕松的。 KISS原理搖滾!

Marshaling時自定義日歷和日期

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

第4章:輸出

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

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

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM