java.util.Date和XMLGregorianCalendar之间的简单转换
我正在寻找一种在java.util.Date和javax.xml.datatype.XMLGregorianCalendar之间双向转换的简单方法 以下是我现在使用的代码:java.util.Date和XMLGregorianCalendar之间的简单转换,java,xml,datetime,jaxb,gregorian-calendar,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.
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类是从模式自动生成的。我的项目上的构建过程不允许我对生成的类进行手动更改。XJC将xs:dateTime元素生成为JAXB类中的XMLGregorianCalendar。模式会定期进行扩展和调整,因此我可以对模式XSD文件进行有限的更改
解决方案更新:
Blaise提出的解决方案允许我将XMLGregorianCalendar从混合中去掉,转而处理java.util.Calendar对象。通过在模式文件的顶部添加JAXB绑定子句,XJC能够为JAXB类中的xs:dateTime生成更合适的映射。下面是一些显示我的XSD文件中的修改的代码片段
XSD文件中的根元素:
JAXB绑定注释块,直接插入XSD中的根元素之后:
因为xmlxs:dateTime字段也存储时区,所以我最好使用日历而不是日期,因为日历对象有一个非常好的API来处理区域设置和时区。无论如何,我更乐意处理日历对象而不是XMLGregorianCalendar。不再需要我上面列出的转换方法。我并没有完全到达java.util.Date,但已经足够近了 为什么不使用外部绑定文件告诉XJC生成java.util.Date字段而不是XMLGregorianCalendar
另请参见我必须进行一些更改以使其正常工作,因为在此期间有些事情似乎发生了变化:
- xjc会抱怨我的适配器没有扩展XmlAdapter
- 引入了一些奇怪和不必要的导入(org.w3._2001.xmlschema)
- 显然,在扩展XmlAdapter时,解析方法不能是静态的
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();
}
}
import java.util.Date;
导入javax.xml.bind.DatatypeConverter;
导入javax.xml.bind.annotation.adapters.XmlAdapter;
导入org.joda.time.DateTime;
公共类DateAdapter扩展了XmlAdapter{
@凌驾
公共对象封送处理(对象dt)引发异常{
返回新的DateTime((Date)dt).toString(“YYYY-MM-dd”);
}
@凌驾
公共对象解组(对象s)引发异常{
返回DatatypeConverter.parseDate((字符串)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>
我也有这种头痛。 在我的POJO中,通过简单地将时间域表示为原始长来摆脱它。 现在,我的WS-client代码的生成正确地处理了一切,不再是XML到Java的废话。当然,在Java端处理millis是简单而无痛的。
亲吻主岩 从XMLGregorianCalendar到java.util.Date,您只需执行以下操作:
java.util.Date dt = xmlGregorianCalendarInstance.toGregorianCalendar().getTime();
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);
从java.util.Date到XMLGregorianCalendar,您只需执行以下操作:
java.util.Date dt = xmlGregorianCalendarInstance.toGregorianCalendar().getTime();
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的第一条注释后编辑的代码,因为我犯了一个错误。在封送处理时自定义日历和日期 步骤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 4:输出
<xmlHeader>
<creationTime>2014-09-25T07:23:05</creationTime> Calendar class formatted
<fileDate>2014-09-25</fileDate> - Date class formatted
</xmlHeader>
2014-09-25T07:23:05日历类已格式化
2014-09-25-日期类格式
您可以使用此自定义项将默认映射更改为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>
我将对此进行调查。谢谢,没问题。JAXB可以处理java.util.Date类型,您只需要在模型中生成它。这可能会很棘手,这对我来说很有效。有关我所做的操作的详细信息,请参阅对上面问题的编辑。我添加了jaxb绑定,但就在xs:schema的正下方,出现了以下错误:com.sun.istack.SAXParseException2:编译器无法执行此globalBindings自定义。它附加到错误的位置,或者与其他绑定不一致。位于..@pritam的com.sun.tools.xjc.ErrorReceiver.error(ErrorReceiver.java:86)-下面是另一个可能有帮助的示例:。对于你所看到的问题,也许值得开始一个新的问题。我不知道任何问题。但是你的似乎很好——只要把它放在
util
包中并使用它就行了?他们有点恼人。如果它们来自jaxb,则可以使用@XMLTypeAdapter直接绑定到java.util.Date。当然,如果您是根据架构自动生成的,那么当您重新生成时,更改对象可能同样令人恼火。@aff我是根据架构自动生成的,因此我不能对生成的JAXB类进行任何手动更改,这与?@Jacob-不是一样。他已经知道了怎么做,他想知道是否没有现成的实用程序类。自从我提出这个问题以来,我就成了Joda Time的粉丝。比JavaSE日期和时间类好得多。非常适合处理时区!谢谢我正在寻找一种将XMLGregorianCalendar转换为毫秒时间的方法。与您编写的方法不同:GregorianCalendar.setTime()不会返回任何内容。
<xmlHeader>
<creationTime>2014-09-25T07:23:05</creationTime> Calendar class formatted
<fileDate>2014-09-25</fileDate> - Date class formatted
</xmlHeader>
<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>