java.util.Date和XMLGregorianCalendar之间的简单转换

Jim*_*ugh 110 java xml datetime jaxb gregorian-calendar

我正在寻找一种在两个方向上在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();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

有没有更简单的东西,比如我忽略了一些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">
Run Code Online (Sandbox Code Playgroud)

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>
Run Code Online (Sandbox Code Playgroud)

由于XML xs:dateTime字段也存储时区,因此我可能更好地使用Calendar而不是Date,因为Calendar对象具有用于处理区域设置和时区的非常好的API.无论如何,我更乐意处理Calendar对象而不是XMLGregorianCalendar.不再需要我上面列出的转换方法了.我没有一路到java.util.Date,但足够接近!

Zé *_*los 81

从XMLGregorianCalendar到java.util.Date,你可以简单地做到:

java.util.Date dt = xmlGregorianCalendarInstance.toGregorianCalendar().getTime();  
Run Code Online (Sandbox Code Playgroud)


bdo*_*han 46

为什么不使用外部绑定文件来告诉XJC生成java.util.Date字段而不是XMLGregorianCalendar?

另请参阅 如何将xs:date映射到java.util.Date?博客


mor*_*bio 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);
Run Code Online (Sandbox Code Playgroud)

在@ f-puras的第一个评论之后编辑的代码,因为我做错了.


Gre*_*gor 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();
    }
}
Run Code Online (Sandbox Code Playgroud)

<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>
Run Code Online (Sandbox Code Playgroud)

在xsd中,我遵循了上面给出的优秀参考,所以我已经包含了这个xml注释:

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();
    }
}
Run Code Online (Sandbox Code Playgroud)