时区之间的转换会产生错误的结果

时间:2015-04-28 07:13:10

标签: java datetime time jodatime

 DateTime dt = new LocalDateTime(rs.getTimestamp("LastUpdated").getTime()).toDateTime(DateTimeZone.forID("America/New_York")); 

Timestamp t=new Timestamp(dt.getMillis());     

我从中得到的结果与“America / New_York”的实际时间不符。这是几个小时。可能是什么问题?

更新 -

这是我要转换的时间来自DB:2015-04-28 13:14:31这是转换后的时间(从代码中删除“dt.plusMillis(offset)”后的代码输出:2015-04-28 22:44:31.0这是纽约的实际时间:凌晨3点45分

2 个答案:

答案 0 :(得分:1)

使用JODA库很容易制作它。核心java库很痛苦,不建议使用它们。但是,日期和时间库已使用Java 8重新实现。

让我们举个例子,假设您尝试将澳大利亚墨尔本时间转换为马德里时间。

这是Joda代码;

    LocalDateTime dateTime = new LocalDateTime(2015, 3, 28, 14, 0, 0); 
    DateTime srcDateTime = dateTime.toDateTime(DateTimeZone.forID("Asia/Colombo"));
    DateTime dstDateTime = srcDateTime.withZone(DateTimeZone.forID("America/New_York"));
    Date madridTime =dstDateTime.toLocalDateTime().toDateTime().toDate();


    System.out.println("Asia/Colombo -> "+srcDateTime.toDateTime());
    System.out.println("America/New_York -> "+madridTime);

这是输出:

Asia/Colombo -> 2015-04-28T14:00:00.050+05:30
America/New_York -> Tue Apr 28 04:30:00 EEST 2015

您也可以从http://www.timezoneconverter.com/

查看

enter image description here

另见:

Date and Time API for JDK 8

答案 1 :(得分:1)

tl; dr

myResultSet
.getObject( … , OffsetDateTime.class )   // Returns a `OffsetDateTime` object represent a moment as seen through some offset-from-UTC, a number of hours-minutes-seconds ahead/behind UTC. 
.atZoneSameInstant(                      // Adjusts from a moment with a mere offset to a moment as seen in a time zone.
    ZoneId.of( "America/New_York" )
)

Joda-Time 处于维护模式

仅供参考,Joda-Time项目现在位于maintenance mode中,建议迁移到java.time类。

LocalDateTime不是片刻

LocalDateTime类(在Joda-Time和 java.time 中)都 not 表示时刻,是 not 时间轴上的一点。此类通常不在业务应用程序中使用,除了(a)将来的约会,以及(b)表示日期的日期可能适用于任何地点,但不适用于特定地点地方。

避免使用java.sql.Timestamp

显然,您是从java.sql.Timestamp检索到ResultSet对象的。别这样

OffsetDateTime

这个可怕的类在几年前被 java.time 类所取代,特别是与OffsetDateTime 4.2和更高版本一起使用时,JDBC被替代。

OffsetDateTime odt = myResultSet.getObject( … , OffsetDateTime.class ) ;

ZonedDateTime

要查看此moment in the New York地区的时区,请应用ZoneId以获取ZonedDateTime对象。

ZoneId z = ZoneId.of( "America/New_York" ) ;
ZonedDateTime zdt = odt.atZoneSameInstant( z ) ;

相同时刻,时间轴上的同时点,不同的时钟时间。

请注意偏移量和区域之间的区别:

  • Offset-from-UTC
    只需几小时-分钟-秒。对于在UTC之前(朝东方)领先,对于在UTC后面(朝西方)落后,负面。
  • Time zone
    时区要远得多。时区是过去,现在和将来(计划的)特定区域的人们使用的偏移量的历史记录。

Table of date-time types in Java (both legacy and modern) and in standard SQL.


关于 java.time

java.time框架已内置在Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendarSimpleDateFormat

要了解更多信息,请参见Oracle Tutorial。并在Stack Overflow中搜索许多示例和说明。规格为JSR 310

目前位于Joda-Timemaintenance mode项目建议迁移到java.time类。

您可以直接与数据库交换 java.time 对象。使用符合JDBC driver或更高版本的JDBC 4.2。不需要字符串,不需要java.sql.*类。

在哪里获取java.time类?

ThreeTen-Extra项目使用其他类扩展了java.time。该项目为将来可能在java.time中添加内容提供了一个试验场。您可能会在这里找到一些有用的类,例如IntervalYearWeekYearQuartermore