Java日期和MySQL时间戳时区

时间:2019-01-23 15:51:19

标签: java mysql jdbc timestamp

我正在编辑一段基本上可以完成的代码:

timestamp = new Date();

然后将timestamp变量保留在TIMESTAMP MySQL表列中。
但是,通过调试时,我看到Date对象显示在正确的时区 GMT + 1 中,当它保留在数据库中时,它是一个 GMT 时间,区,所以一个小时前。

使用函数CURRENT_TIMESTAMP返回GMT + 1日期。
连接字符串为jdbc:mysql://..../db-name,没有任何参数。

编辑: 找到了这段代码

preparedStatement.setTimestamp(2, new Timestamp(timestamp.getTime()));

发生了什么事?

1 个答案:

答案 0 :(得分:1)

tl; dr

myPreparedStatement
.setObject(  
    … ,                                   // Specify which placeholder `?` in your SQL statement. 
    OffsetDateTime.now( ZoneOffset.UTC )  // Capture the current moment as seen in the wall-clock time of UTC (an offset-from-UTC of zero).
) ;

避免使用旧的日期时间类

您使用的是可怕的日期时间类,而这些类早已被 java.time 类所取代。

请勿使用DateTimestamp

UTC

以UTC捕获当前时刻。大多数数据库都会在UTC中存储时间。通常,您应该在UTC中执行大多数业务逻辑,调试,日志记录,存储和数据交换。

OffsetDateTime

使用恰当命名的OffsetDateTime类表示具有UTC偏移量的时刻。

我们需要UTC本身,或者是零偏移量。为此,我们可以使用一个常量ZoneOffset.UTC

OffsetDateTime odt = OffsetDateTime.now( ZoneOffset.UTC ) ) ;

JDBC 4.2

从JDBC 4.2开始,我们可以直接与数据库交换 java.time 对象。

要将这一刻保存到类似于SQL标准TIMESTAMP WITH TIME ZONE的数据类型的列中:

myPreparedStatement.setObject( … , odt ) ;

检索:

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

ZonedDateTime

要向用户显示此检索的时刻,您可能需要调整到用户的预期/期望时区。

ZoneId z = ZoneId.of( "Africa/Tunis" ) ;
ZonedDateTime zdt = odt.atZoneSameInstant( z ) ;

从不依赖默认时区

在上面的代码中,请注意,我们总是指定所需/期望的偏移量或区域。

如果未指定,则将隐式应用偏移或区域。最好明确地指定您的意图,因为作为程序员,JVM,数据库和主机OS的当前默认值都不在您的手中。这意味着依赖默认值的代码在运行时的行为会有所不同。

Java 6和7

  

但是我仍然被迫处理Java 6

领导JSR 310和 java.time 实现的同一个人Stephen Colebourne,以及著名的 Joda-Time 项目,也领导了另一个项目, ThreeTen-Backport。大部分 java.time 功能都通过几乎相同的API反向移植到该库中的Java 6和7。

您在后端口类中所做的所有工作也是如此。然后,在最后一刻,通过DateTimeUtils类与java.sql.Timestamp进行相互转换。

这些转换方法大多使用Instant对象。 Instant是UTC中的时刻,始终是UTC中的时刻。您可以通过提取OffsetDateTimeInstant调整为UTC。 Instant类是 java.time 中的基本构建块类,其中OffsetDateDate具有更大的灵活性,例如在生成字符串时可以使用其他格式。但是InstantOffsetDateTime都代表一个时刻,即时间轴上的一个点。

Instant instant = odt.toInstant() ;  
java.sql.Timestamp ts = org.threeten.bp.DateTimeUtils.toSqlTimestamp( instant ) ;

朝另一个方向,从数据库中检索Timestamp,然后立即转换为Instant

java.sql.Timestamp ts = myResultSet.getTimestamp( … ) ;
Instant instant = org.threeten.bp.DateTimeUtils.toInstant( ts ) ;

关于 java.time

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

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

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

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

在哪里获取java.time类?

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