将一个时区的日期时间字符串转换为另一个时区的日期时间字符串,使用偏移量。

7
我有一个日期时间字符串"2018-01-15 01:16:00",它在东部标准时间区。我想使用UTC偏移量将其动态转换为另一个时区。我的javascript代码将此UTC偏移量作为参数传递,servlet必须将此日期时间字符串转换/格式化为由提供的偏移量所确定的时区。
我尝试了许多方法,包括在oracle tutorials中记录的方法,但无法得出解决方案。
以下是我正在尝试的代码,非常感谢您的帮助。
private static final String DATE_FORMAT = "yyyy-MM-dd HH:mm:ss";
private static final String DEFAULT_TIME_ZONE = ZoneId.SHORT_IDS.get("EST");

public static void main(String[] args) throws Exception {
    String dateTime = "2018-01-15 02:35:00";
    //parse the datetime using LocalDateTime
    LocalDateTime defaultDateTime = LocalDateTime.parse(dateTime, DateTimeFormatter.ofPattern(DATE_FORMAT));

    //get the datetime in default timezone
    ZoneId defaultZoneId = ZoneId.of(DEFAULT_TIME_ZONE);
    ZonedDateTime defaultZoneDateTime = defaultDateTime.atZone(defaultZoneId);
    System.out.println("EST time: "+defaultZoneDateTime.format(DateTimeFormatter.ofPattern(DATE_FORMAT)));
    ZonedDateTime utcZonedDateTime = defaultZoneDateTime.withZoneSameInstant(ZoneId.of("UTC"));
    String utcTime = defaultZoneDateTime.withZoneSameInstant(ZoneId.of("UTC")).format(DateTimeFormatter.ofPattern(DATE_FORMAT));
    System.out.println("UTC : "+utcTime);

    //IST timezone
    ZoneOffset offset = ZoneOffset.of("+05:30");
    OffsetDateTime offsetDate = OffsetDateTime.of(utcZonedDateTime.toLocalDateTime(), offset);
    String targetTimeZone = offsetDate.format(DateTimeFormatter.ofPattern(DATE_FORMAT));
    System.out.printf("target time : "+targetTimeZone);

}

输出

EST time: 2018-01-15 02:35:00
UTC : 2018-01-15 07:37:00
target time : 2018-01-15 07:37:00

预计目标时间: 2018-01-15 13:05:00


2
首先需要检查的是,“EST”这个词汇到底指的是什么?你是指“东部时间”(根据一年中的时间而定,为UTC+5或UTC+6),还是真正意义上的“东部标准时间”(全年都为UTC+5)?更好的做法是始终使用完整的时区ID,例如“America/New_York”。 - Jon Skeet
接下来:提供一个完整的示例和实际输出是很好的,但如果您也能提供预期输出,那将更有帮助。 - Jon Skeet
谢谢,我的意思是真正的“东部标准时间”,因为这个日期时间字符串来自第三方API,不提供准确的信息。 - Rakesh
感谢 @JonSkeet,我已经添加了期望的输出。我发布的代码是一个带有硬编码输入日期时间字符串的完整示例。 - Rakesh
1
我的早期评论应该提到UTC-5和UTC-4而不是UTC+5和UTC+6。 - Jon Skeet
2个回答

12

直接问题是这一行:

OffsetDateTime offsetDate = OffsetDateTime.of(utcZonedDateTime.toLocalDateTime(), offset);

这句话的意思是你想要相同 本地 的日期/时间,但使用指定的偏移量。这会改变所表示的时间点。

相反,你真正想要的是在特定的偏移量下表示相同的 时间点。因此,最简单的修复方法是:

OffsetDateTime offsetDate = utcZonedDateTime.toInstant().atOffset(offset);

然而,还有其他几个方面需要改变:

  • 优先使用 ZoneOffset.UTC 而不是 ZoneId.of("UTC")
  • 使用 EST 作为时区会导致混淆——不清楚您是否希望它表示“东部时间”(在 EST 和 EDT 之间变化)还是 UTC-5 的纯标准时间。 如果您实际上是指“东部时间”,最好使用 America/New_York 作为区域 ID。
  • 如果输入字符串代表东部时间中跳过或模糊的值,则不清楚您希望发生什么情况。 这些情况发生在夏令时转换期间。

接下来,您根本不需要将东部时间中的 ZonedDateTime 转换为 UTC 中的 ZonedDateTime。 直接将其转换为一个瞬间:

OffsetDateTime target = defaultZoneDateTime.toInstant().at(offset);

或者为目标创建一个ZonedDateTime

ZonedDateTime target = defaultZoneDateTime.withZoneSameInstant(offset);

考虑到偏移量并不是一个真正的时区,我可能会选择这些选项中的第一个。


5
您正在使用的是:

OffsetDateTime.of(utcZonedDateTime.toLocalDateTime(), offset)

创建你的目标。因此,你正在构建一个OffsetDateTime在目标偏移量中,它的LocalDateTime等于UTC区域中的LocalDateTime。

你所想要的正是与你从东部标准时间到UTC使用的转换完全相同:保持相同的时刻,但进入不同的时区:

defaultZoneDateTime.withZoneSameInstant(offset);

如果您真的需要一个OffsetDateTime而不是ZonedDateTime:

OffsetDateTime.ofInstant(defaultZoneDateTime.toInstant(), offset);

1
可能只是我的个人喜好,但要获取 OffsetDateTime,我会选择 defaultZoneDateTime.withZoneSameInstant(offset).toOffsetDateTime() - Ole V.V.

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接