当前位置: 首页 > 知识库问答 >
问题:

Zulu和Offset的日期时间格式

年健
2023-03-14

我希望有一个单一的日期时间模式表达式能够:

  • 使用偏移指示器序列化日期/瞬间,例如:2017-07-13T21:20:33.123 0000

重要的一点是序列化始终包括偏移量(0000),反序列化可以处理这两种情况(Z0000

到目前为止,我所尝试的:

我正在使用JDK8,带有新的内置的java.time包。我尝试创建一个java.time.DateTimeFor事具有以下模式:

yyyy-MM-dd'HH:MM:ss。SSSXX将用于解析Z0000,但在序列化时不起作用。(例如使用UTC时,仅输出Z

yyyy-MM-dd'HH:MM:ss。SSSZ会解析0000,但不会解析Z(如果存在)

这有可能有一个表达式吗?

下面是完全可测试的代码:

import java.time.Instant;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;

public class DateTimeFormatterTest {

    public static void main(String[] args) {
        final DateTimeFormatter dtf = DateTimeFormatter
            .ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSSZ")
            .withZone(ZoneId.of("UTC"));
        final Instant now = Instant.now();

        System.out.println(dtf.format(now)); //THIS IS OK

        final String dateStringOffset = "2017-07-13T21:20:33.123+0200";
        System.out.println(dtf.parse(dateStringOffset)); //THIS IS OK

        final String dateStringZulu = "2017-07-13T21:20:33.123Z";
        System.out.println(dtf.parse(dateStringZulu)); //THIS IS NOT OK
    }
}

我的印象是,将XX附加到模式的末尾会迫使它总是以0000的格式输出偏移量。

共有2个答案

马奇略
2023-03-14
DateTimeFormatter formatter = new DateTimeFormatterBuilder()
    .append(ISO_LOCAL_DATE_TIME)
    .appendPattern("[x][X]")
    .toFormatter();

System.out.println(formatter.parse("2017-07-13T21:20:33.123+0200")
    .get(ChronoField.YEAR)); // 2017
System.out.println(formatter.parse("2017-07-13T21:20:33.123Z")
    .get(ChronoField.YEAR)); // 2017

编辑:

可以使用以下方法进行格式设置:

DateTimeFormatter。模式(“yyyy-MM-dd'T'HH:MM:ss.SSSxx”)

毋炳
2023-03-14

我实际上已经找到了一种方法,但这不是一个漂亮的代码(它使用反射,如果可能的话,我个人更喜欢不使用反射)。

我同意这些评论,有两个格式化程序(一个用于解析,另一个用于格式化)要容易得多。

基本上是因为它似乎没有办法只用一个格式化程序来完成它。我尝试了不同的可选模式组合,最接近的是:

DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSS[xx][XX]").withZone(ZoneOffset.UTC);

这解析了偏移量和祖鲁,但是在格式化时,输出以0000Z结束(因为格式化程序在格式化时总是打印所有可选部分——它似乎没有办法改变这种行为)。

我能做到的唯一方法是反思。首先,我创建一个ZoneId,然后将ID更改为空的String,并将规则与UTC匹配(因此它的行为就像是UTC)。然后我在格式化程序中使用这个修改过的区域:

// get any zone, just to get a valid ZoneId
ZoneId zone = ZoneId.of("Europe/London");
// change ID to empty string (so it's not printed by the formatter)
Field field = zone.getClass().getDeclaredField("id");
field.setAccessible(true);
field.set(zone, "");
// change zone rules to match UTC (so this zone becomes a "copy" of UTC)
field = zone.getClass().getDeclaredField("rules");
field.setAccessible(true);
field.set(zone, ZoneOffset.UTC.getRules());

DateTimeFormatter dtf = new DateTimeFormatterBuilder()
    // date and time
    .appendPattern("yyyy-MM-dd'T'HH:mm:ss.SSS")
    // optional offset - prints +0000 when it's zero (instead of Z)
    .optionalStart().appendOffset("+HHMM", "+0000").optionalEnd()
    // optional zone id (so it parses "Z")
    .optionalStart()
    .appendZoneId()
    // add default value for offset seconds when field is not present
    .parseDefaulting(ChronoField.OFFSET_SECONDS, 0)
    .optionalEnd()
    // create formatter using the "UTC-cloned" zone
    .toFormatter().withZone(zone);

System.out.println(dtf.format(ZonedDateTime.now()));
System.out.println(dtf.format(Instant.now()));

String dateStringOffset = "2017-07-13T21:20:33.123+0200";
System.out.println(dtf.parse(dateStringOffset));

String dateStringZulu = "2017-07-13T21:20:33.123Z";
System.out.println(dtf.parse(dateStringZulu));

输出为:

2017-07-12T13:39:49.695 0000
2017-07-12T13:39:49.695 0000
{OffsetSeconds=7200,InstantSeconds=14999980833},ISO,解析为2017-07-13T21:20:33.123
{OffsetSeconds=0,InstantSeconds=14999980833},ISO,Z解析为2017-07-13T21:20:33.123

我必须添加parseDefaulting(ChronoField.OFFSET_SECONDS,0),因为当找到Z时,没有设置OffsetSeconds字段,结果不能用于创建OffsetDateTime

这个格式化程序存在一些问题:

  • 首先是反射的使用,当然。在正式生产环境中使用它是一个选择问题(我个人避免使用它),如果将SecurityManager配置为不允许它(以及其他一些问题),它可以抛出一个SecurityExc0019
  • 更微妙和棘手的细节是被解析的对象更喜欢而不是偏移量。

因此,在第一种情况下(dateStringOffset=2017-07-13T21:20:33.123 0200),如果我尝试创建一个InstantZoneDateTime,它使用本地时间21:20:33和区域作为UTC,忽略偏移0200

dtf.parse(dateStringOffset, Instant::from); // Wrong: 2017-07-13T21:20:33.123Z
dtf.parse(dateStringOffset, ZonedDateTime::from); // Wrong: 2017-07-13T21:20:33.123Z[]

为了得到正确的结果,我必须使用OffsetDateTime

dtf.parse(dateStringOffset, OffsetDateTime::from); // Correct: 2017-07-13T21:20:33.123+02:00

对于第二种情况,在UTC(Z),这三种类型都可以工作。但我建议始终将结果解析为OffsetDateTime,然后将其转换为其他类型。

或者保持简单,只需使用2个格式化程序:

// parse both offset and Z
DateTimeFormatter parser = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSS[xx][XX]");
// format with offset (and +0000 instead of Z)
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSSxx").withZone(ZoneOffset.UTC);

这没有上述副作用-它正确解析偏移量:

parser.parse(dateStringOffset, Instant::from); // 2017-07-13T19:20:33.123Z (Instant is always in UTC)
parser.parse(dateStringOffset, ZonedDateTime::from); // 2017-07-13T21:20:33.123+02:00
parser.parse(dateStringOffset, OffsetDateTime::from); // 2017-07-13T21:20:33.123+02:00
 类似资料:
  • 下表显示可用于为图表中的日期时间字段创建用户定义的数据格式的说明符。 说明符 描述 D 将日显示为不带前导零的数字(1-31)。 DD 将日显示为带前导零的数字(01-31)。 M 将月份显示为不带前导零的数字(1-12)。 MM 将月份显示为带前导零的数字(01-12)。 MMM 将月份显示为缩写形式(Jan-Dec)。 MMMM 将月份显示为完整月份名(January-December)。 Y

  • 下表显示可用于为图表中的日期时间字段创建用户定义的数据格式的说明符。 说明符 描述 D 将日显示为不带前导零的数字(1-31)。 DD 将日显示为带前导零的数字(01-31)。 M 将月份显示为不带前导零的数字(1-12)。 MM 将月份显示为带前导零的数字(01-12)。 MMM 将月份显示为缩写形式(Jan-Dec)。 MMMM 将月份显示为完整月份名(January-December)。 Y

  • 下表显示可用于为图表中的日期时间字段创建用户定义的数据格式的说明符。 说明符 描述 D 将日显示为不带前导零的数字(1-31)。 DD 将日显示为带前导零的数字(01-31)。 M 将月份显示为不带前导零的数字(1-12)。 MM 将月份显示为带前导零的数字(01-12)。 MMM 将月份显示为缩写形式(Jan-Dec)。 MMMM 将月份显示为完整月份名(January-December)。 Y

  • 问题内容: 为什么python 2.7不像JavaScript那样在UTC日期时间对象的isoformat字符串的末尾不包含Z字符(Zulu或零偏移)? 而在javascript中 问题答案: Python对象默认没有时区信息,没有它,Python实际上违反了ISO 8601规范(如果未提供时区信息,则假定为本地时间)。您可以使用pytz包获取一些默认时区,或者直接自己子类化: 然后,您可以将时区

  • 问题内容: 我需要改变从一个日期/时间 2014年8月20日15:30:00 看起来像 二○一四年八月二十○日下午3:30 可以使用javascript的Date对象完成此操作吗? 问题答案: 是的,您可以使用本地javascript Date() 对象 及其方法。 例如,您可以创建一个类似以下的函数: 并显示上午/下午和正确的时间。 请记住使用 getFullYear() 方法,而不要使用get

  • 您如何创建一个函数来将之类的内容转换为使用Dart的日期和时间,例如和?