首页 > 解决方案 > 如何从给定的字符串转换时区

问题描述

我有一个给定的字符串06/17/2008T13:53:23Z,我想将其转换EDT为时区。所以我的输出将是6/17/08 9:53 AM EDT.

下面是我的 Java 代码,它给出了错误的输出:

    SimpleDateFormat format1 = new SimpleDateFormat("MM/dd/yyyy'T'hh:mm:ss'Z'");
    SimpleDateFormat format2 = new SimpleDateFormat("M/d/yy hh:mm a");
    format1.setTimeZone(TimeZone.getTimeZone("UTC"));
    Date date = format1.parse("06/17/2008T13:53:23Z");
    System.out.println(date);
    format2.setTimeZone(TimeZone.getTimeZone("EDT"));
    System.out.println(format2.format(date));

它给出的输出为:

Tue Jun 17 19:23:23 IST 2008
6/17/08 01:53 PM

这里的输出时间01:53 PM9:53 AM EDT

如何解决这个问题?

标签: javatimetimezone

解决方案


您的解析模式使用hh(用于 12 小时格式),而日期时间字符串中的时间是 24 小时格式,您需要使用HH. 其次,您应该避免使用三个字母的时区名称。正如您已经预料到的那样,EDT有一个-4小时的区域偏移量,您可以将其GMT-4SimpleDateFormat.

虽然 Zone-Offset 以数字(小时、分钟或秒)表示,但时区表示为表示时区名称(例如America/New_York)的字符串。timezone 和 Zone-Offset 之间的关系是多对一的,即多个时区可以具有相同的 Zone-Offset。

请注意,java.util日期时间类已过时且容易出错,它们的格式化 API 也是如此,SimpleDateFormat. 我建议您应该完全停止使用它们并切换到现代日期时间 API在Trail: Date Time了解有关现代日期时间 API 的更多信息。

如果您正在为您的 Android 项目执行此操作,并且您的 Android API 级别仍然不符合 Java-8,请通过 desugaring和How to use ThreeTenABP in Android Project检查Java 8+ APIs available

使用现代日期时间 API:

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

public class Main {
    public static void main(String[] args) {
        String str = "06/17/2008T13:53:23Z";
        DateTimeFormatter formatter = DateTimeFormatter.ofPattern("M/d/u'T'H:m:sz");
        ZonedDateTime zdt = ZonedDateTime.parse(str, formatter);

        // Convert to Eastern Time
        ZonedDateTime zdtET = zdt.withZoneSameInstant(ZoneId.of("America/New_York"));
        // ZonedDateTime zdtET = zdt.withZoneSameInstant(ZoneOffset.ofHours(-4));// Or this
        
        // Print in default format i.e. ZonedDateTime#toString
        System.out.println(zdtET);

        // Print in custom formats
        System.out.println(zdtET.format(DateTimeFormatter.ofPattern("MM/dd/uuuu'T'HH:mm:ss zzzz")));
        System.out.println(zdtET.format(DateTimeFormatter.ofPattern("M/d/uu hh:mm:ss a")));
    }
}

输出:

2008-06-17T09:53:23-04:00[America/New_York]
06/17/2008T09:53:23 Eastern Daylight Time
6/17/08 09:53:23 am

注意:如果您使用ZoneOffset.ofHours(-4)[上面代码中的注释],则无法America/New_York在输出中获取时区名称(例如 ),因为如前所述,许多时区可以具有相同的 Zone-Offset 并且没有默认时区区域偏移。

使用旧版 API:

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.TimeZone;

public class Main {
    public static void main(String[] args) throws ParseException {
        String str = "06/17/2008T13:53:23Z";
        SimpleDateFormat formatter = new SimpleDateFormat("MM/dd/yyyy'T'HH:mm:ss");
        formatter.setTimeZone(TimeZone.getTimeZone("GMT+0"));
        Date date = formatter.parse(str);

        // Convert to Eastern Time
        SimpleDateFormat sdfOutput = new SimpleDateFormat("M/d/yy hh:mm a");
        sdfOutput.setTimeZone(TimeZone.getTimeZone("America/New_York"));
        // sdfOutput.setTimeZone(TimeZone.getTimeZone("GMT-4"));// Or this

        System.out.println(sdfOutput.format(date));
    }
}

输出:

6/17/08 09:53 am

推荐阅读