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

【问题讨论】:

我建议你不要使用SimpleDateFormatDate。这些类设计不佳且早已过时,尤其是前者,尤其是出了名的麻烦。而是使用来自java.time, the modern Java date and time API 的LocalDateTimeDateTimeFormatter 类似: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的主要内容,如果未能解决你的问题,请参考以下文章

Java 日期格式转 JavaScript 日期格式

格式为“yyyy-mm-dd HH:mm:ss”的 Java 字符串到日期对象

Java中关于时间日期格式保存到mysql的问题

通过匹配字符串中的日期格式使用Java提取日期

如何检查字符串是不是匹配某种日期格式?? JAVA [重复]

更改 Java 字符串中的日期格式