Java 日期格式:字符串到 XMLGregorianCalendar
Posted
技术标签:
【中文标题】Java 日期格式:字符串到 XMLGregorianCalendar【英文标题】:Java Date format : String to XMLGregorianCalendar 【发布时间】:2021-08-15 17:18:28 【问题描述】:我有一个字符串日期,我将其转换为XMLGregorianCalendar
。下面的代码将其转换为没有错误。但是,我将日期传递给的系统会收到下面报告的错误。
以下XMLGregorianCalendar
有效:
下面的XMLGregorianCalendar
不起作用(下面的代码将它转换成这个):
问题
如何将字符串转换为所需的格式?
public static void main(String []args)
try
String commenceTOString = "2021-06-30 05:00:00";
java.util.Date dateCommence = new java.text.SimpleDateFormat("yyyy-MM-dd HH:mm:ss").parse(commenceTOString);
System.out.println(dateCommence);
java.util.GregorianCalendar c = new java.util.GregorianCalendar();
c.setTime(dateCommence);
javax.xml.datatype.XMLGregorianCalendar commence = javax.xml.datatype.DatatypeFactory.newInstance().newXMLGregorianCalendar(c);
System.out.println(commence);
catch (Exception e)
e.printStackTrace();
错误
E:消息验证失败。错误:[cvc-pattern-valid:值 '2021-06-30T06:30:00.000+02:00' 不是方面有效的 图案 '(((((0[1-9])|(1[0-2]))-((0[1-9])|(1\d)|(2[0-9]))) |((((0[13578])|(1[02]))-31)|(((0[1,3-9])|(1[0-2]))-30)))| (((19|20)(([02468][048])|([13579][26]))-02-29))|((20[0-9][0-9])|(19 [0-9][0-9]))-((((0[1-9])|(1[0-2]))-((0[1-9])|(1\d) |(2[0-8])))|((((0[13578])|(1[02]))-31)|(((0[1,3-9])|(1[0 -2]))-(29|30)))))T([0-1][0-9]|[2][0-3])(:([0-5][0-9] ))1,2' 对于类型“dateTime”.cvc-attribute.3:值 属性“DepartureDateTime”的“2021-06-30T06:30:00.000+02:00”开启 元素“FlightSegment”就其类型而言无效, '约会时间'。]。 http://www.elleipsis.com/booking/exception/35a8ac87-6743-4719-9504-a8a17a1a0471
【问题讨论】:
我建议你不要使用SimpleDateFormat
和Date
。这些类设计不佳且早已过时,尤其是前者,尤其是出了名的麻烦。而是使用来自java.time, the modern Java date and time API 的LocalDateTime
和DateTimeFormatter
。
类似:XMLGregorianCalendar in java, with NO Timezone.
【参考方案1】:
有很多方法可以做到这一点。一般来说,我建议您使用 java.time 进行日期和时间工作,当 java.time 类和 XMLGregorianCalendar
之间没有直接转换时也是如此。
破解
hack 只是将T
编辑成XMLGregorianCalendar
期望存在的字符串,然后直接解析修改后的字符串。那么我们既不需要java.time,也不需要Date
,也不需要SimpleDateFormat
。
String commenceTOString = "2021-06-30 05:00:00";
XMLGregorianCalendar commence = DatatypeFactory.newInstance()
.newXMLGregorianCalendar(commenceTOString.replace(' ', 'T'));
System.out.println(commence);
输出:
2021-06-30T05:00:00
通过字符串转换
解析成LocalDateTime
,格式化成XMLGregorianCalendar
格式的字符串,让DatatypeFactory
再次解析:
DateTimeFormatter parser = DateTimeFormatter.ofPattern("uuuu-MM-dd HH:mm:ss");
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("uuuu-MM-dd'T'HH:mm:ss");
LocalDateTime dateTime = LocalDateTime.parse(commenceTOString, parser);
XMLGregorianCalendar commence = DatatypeFactory.newInstance()
.newXMLGregorianCalendar(dateTime.format(formatter));
System.out.println(commence);
输出和以前一样。
通过数字转换
虽然长了几行,但这是概念上简单的转换:将每个字段从 LocalDateTime
转移到 XMLGregorianCalendar
。
XMLGregorianCalendar commence = DatatypeFactory.newInstance()
.newXMLGregorianCalendar(
dateTime.getYear(), dateTime.getMonthValue(), dateTime.getDayOfMonth(),
dateTime.getHour(), dateTime.getMinute(), dateTime.getSecond(),
DatatypeConstants.FIELD_UNDEFINED, DatatypeConstants.FIELD_UNDEFINED);
System.out.println(commence);
输出还是一样的。
通过公历转换
我相信这是官方的转换:转换为ZonedDateTime
转换为GregorianCalendar
,最后转换为XMLGregorianCalendar
。我们需要将不需要的字段设置为FIELD_UNDEFINED
,就像您自己的答案一样。
ZonedDateTime dateTime = LocalDateTime.parse(commenceTOString, parser)
.atZone(ZoneOffset.UTC);
GregorianCalendar gc = GregorianCalendar.from(dateTime);
XMLGregorianCalendar commence = DatatypeFactory.newInstance()
.newXMLGregorianCalendar(gc);
commence.setTimezone(DatatypeConstants.FIELD_UNDEFINED);
commence.setMillisecond(DatatypeConstants.FIELD_UNDEFINED);
System.out.println(commence);
链接
Oracle tutorial: Date Time 解释如何使用 java.time。
【讨论】:
【参考方案2】:这为我修复了它:(使用DatatypeConstants.FIELD_UNDEFINED
)
GregorianCalendar c = new GregorianCalendar();
c.setTime(dateCommence);
XMLGregorianCalendar commence = DatatypeFactory.newInstance().newXMLGregorianCalendar(c);
commence.setTimezone(DatatypeConstants.FIELD_UNDEFINED);
commence.setMillisecond(DatatypeConstants.FIELD_UNDEFINED);
【讨论】:
以上是关于Java 日期格式:字符串到 XMLGregorianCalendar的主要内容,如果未能解决你的问题,请参考以下文章
格式为“yyyy-mm-dd HH:mm:ss”的 Java 字符串到日期对象