2

我需要一些关于这个 java 方法的建议。此方法的目的是获取一个表示日期的字符串 - 此字符串是根据 EST 时区中的日期创建的 - 并将其转换为 UTC 时区中的 java Date 对象。

private Date buildValidationDate(String dateString) throws ParseException {
    System.out.println("dateString " + dateString);

    SimpleDateFormat dateFormat = new SimpleDateFormat("MM/dd/yyy hh:mm a");
    dateFormat.setTimeZone(TimeZone.getTimeZone("UTC"));  
    dateFormat.setLenient(true);
    Date dt = dateFormat.parse(dateString);

    System.out.println("dt " + dt);

    return dt;
}

我看到的问题是 dt 的值似乎关闭了。例如,如果 dateString 是 '10/16/2012 12:06 PM' - 我预计 dt 的值(以 UTC 为单位)类似于 'Tuesday, October 16, 2012 4:06 PM'。相反,dt 的值是 'Tue Oct 16 07:06:00 CDT 2012'。这似乎不是正确的 UTC 时间。

我很感激任何建议,如果这似乎是一个简单的问题,我很抱歉我在 Java 日期方面遇到了很多麻烦。我不确定我的编码是否错误,或者我的方法是否有问题。谢谢

4

2 回答 2

4

正如Singh 正确接受的答案所说,您Date实际上在 UTC 中,但它的toString方法在生成字符串时会混淆地应用当前默认时区。

ISO 8601

避免使用日期时间值等格式10/16/2012 12:06 PM。序列化为文本时,请使用为此目的定义为标准的ISO 8601格式。

java.time

如果这似乎是一个简单的问题,我很抱歉我在 Java 日期方面遇到了很多麻烦

不是你;这是课程。旧的遗留日期时间类在处理日期时间方面是行业领先的英勇努力。但事实证明,它们构思不周、设计糟糕、非常混乱和麻烦。现在被 java.time 类所取代——一个巨大的改进。

完全避免这个麻烦的旧java.util.Date课程。而是使用Instant它的位置。

Instant

该类表示UTCInstant时间线上的时刻,分辨率为纳秒(最多九 (9) 位小数)。

获取当前时刻。

Instant instant = Instant.now();

您可以通过调用添加到旧日期时间类的新转换方法之一将 Date 转换为其现代替换。只需调用toInstant,很简单。

Instant instant = myJavaUtilDate.toInstant(); 

java.time 类在生成字符串时默认使用 ISO 8601 格式。只需调用toString即可获得对象内值的清晰表示。

String output = instant.toString();

2016-12-23T01:33:09.731Z

解析

要解析您的输入字符串,请定义要匹配的格式模式。模式代码与 的相似,SimpleDateFormat但不完全相同。所以一定要仔细研究文档。

String input = "10/16/2012 12:06 PM" ;
DateTimeFormatter f = DateTimeFormatter.ofPattern( "MM/dd/uuuu hh:mm a" );

您的输入缺乏关于从 UTC 偏移或时区的任何线索。所以我们必须解析为LocalDateTime. 缺少任何偏移或区域,aLocalDateTime只是关于可能时刻的模糊概念,但并不代表时间线上的一个点。

LocalDateTime ldt = LocalDateTime.parse( input , f );

ldt.toString(): 2012-10-16T12:06

该问题声称这是针对“EST 时区”的。所以我们需要对我们应用一个时区 aZoneIdLocalDateTime获得 a ZonedDateTime

以、或等格式指定适当的时区名称。永远不要使用 3-4 个字母的缩写,例如或,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。continent/regionAmerica/MontrealAfrica/CasablancaPacific/AucklandESTIST

也许EST您指的是美国和加拿大东海岸大部分地区使用的时区。我会随意选择America/New_York

ZoneId z = ZoneId.of( "America/New_York" );
ZonedDateTime zdt = ldt.atZone( z );

zdt.toString(): 2012-10-16T12:06-04:00[美国/纽约]

要获得 UTC,只需提取一个Instant. 您可以在概念上将其视为:

ZonedDateTime = ( Instant + ZoneId )

Instant instant = zdt.toInstant();

即时.toString(): 2012-10-16T16:06:00Z

于 2016-12-23T01:37:58.223 回答
3

您的日期正在正确转换。它只是以您的默认时区格式打印值,与java.util.Date时区无关。如果您想要特定时区的处理,请使用java.util.Calendar.

于 2012-10-16T16:24:38.903 回答