【问题标题】:Serializing java.sql.Date through Jackson while accounting for Daylight Savings Time通过 Jackson 序列化 java.sql.Date,同时考虑夏令时
【发布时间】:2019-07-26 04:26:43
【问题描述】:

我正在使用java.sql.Date 将日期字段存储在我的一个域对象中。此字段映射到 MySQL DATE 列。当我尝试通过 Jackson 将此字段序列化为 JSON 时,Jackson 似乎没有考虑夏令时。

这是我的域对象中该字段的样子:

@JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd", timezone = "EST")
private Date date;

如您所见,Jackson 已被指示将时区解释为 EST。我的 MySQL 数据库也使用 EST 时区:

SHOW VARIABLES LIKE '%zone';
Output:
system_time_zone | EST
time_zone        | SYSTEM

我的问题是,对于夏令时开始和结束之间的日期,Jackson 返回的日期比数据库中存储的日期少一天。我假设这是因为杰克逊没有考虑夏令时。

我通过尝试将 Java 字段中的日期格式从 yyyy-MM-dd 更改为 yyyy-MM-dd HH:mm:ss 得出了这个结论。我注意到服务器返回了类似2017-03-13 00:00:00 的内容,但Jackson 将其序列化为2017-03-12 23:00:00(少了一小时,这是美国东部时间受夏令时影响的时间)。

有什么办法可以解决这个问题吗?另外,考虑到它没有时间对应物,java.sql.Date 是正确使用的类型吗?我一直在考虑改用java.time.LocalDate,但我还没有看到它是否会成功。

提前致谢!

【问题讨论】:

  • 还注意到使用java.time.LocalDate 似乎可以解决问题。不知道为什么。
  • 我认为这不是杰克逊的错,我怀疑您的 MySQL 数据库,或者特别是您用于连接数据库的驱动程序。该驱动程序负责将数据库中的值转换为java.sql.Date,并且它使用(据我所知)您的当前/本地时区。我在 MySQL、标准驱动程序和夏令时也遇到了类似的问题,所以我改用 UTC 来避免所有这些问题。
  • 嗯,从技术上讲,杰克逊也可能受到指责,因为 2017-03-13 00:00:00 在 EST 中不存在,仅在 EDT 中存在,对吧?
  • 真正的责备属于java.sql.Date,因为他在假装只有日期​​的同时携带一天中的时间。
  • @Tom 没错,2017-03-13 00:00:00 是 EDT 时间。正如您所说,这可能是驱动程序问题,切换到 UTC 是有道理的。

标签: java mysql date jackson


【解决方案1】:

tl;博士

  • 切勿使用 java.sql.Datejava.util.Date
    仅使用 java.time 类。
  • 对于仅日期值,使用LocalDate
    您只会看到稳定的值,不受夏令时 (DST) 的影响。

例子:

LocalDate.parse( "2019-01-23" )

java.sql.Date 不是日期

我不确定到底是什么问题,但我怀疑这是由于您使用了糟糕的 java.sql.Date 类,因此没有实际意义。

java.sql.Date假装 表示仅日期值,没有时间和时区。然而,由于一些难以想象的糟糕设计决策,该类扩展了java.util.Datejava.util.Date确实有时间并且是UTC。更令人困惑的是,java.util.Date 在其源代码中隐藏了一个时区,没有 getter 和 setter,因此它似乎还没有影响像 equals 这样的方法的行为。所以java.sql.Date,尽管它的名字,尽管它的目的是只保存一个日期,实际上确实有一个时间设置为UTC。所以java.sql.Date 在调整一天中的时间方面采取了一些技巧,这可能是您遇到的问题。这些传统的日期时间课程是一大堆热气腾腾的……燕麦片。你不应该使用它们。

java.sql.Date 类引用the JavaDoc

一个毫秒值的瘦包装器,允许 JDBC 将其识别为 SQL DATE 值。毫秒值表示自 1970 年 1 月 1 日 00:00:00.000 GMT 以来经过的毫秒数。

为了符合 SQL DATE 的定义,由 java.sql.Date 实例包装的毫秒值必须通过在特定时区中将小时、分钟、秒和毫秒设置为零来进行“规范化”。实例已关联。

顺便说一句,java.sql.Timestamp 也是一团糟。它也笨拙地从 java.util.Date 继承,添加了一个 second 小数秒的纳秒。也避免使用这个类,现在替换为java.time.Instantjava.time.OffsetDateTime。同样,java.sql.Timejava.time.LocalTime 替换。

java.time.LocalDate真正日期

随着 JSR 310 和 JDBC 4.2 的采用,您可以使用现代业界领先的 java.time 类。到目前为止,Jackson 可能已经更新为使用 java.time。如果没有,请参阅this Question 以获取数据类型模块的链接以处理 Jackson 中的 java.time

看起来您的输入字符串是标准的ISO 8601 格式,YYYY-MM-DD。 java.time 类在解析/生成表示日期时间值的字符串时默认使用标准格式。对于仅日期使用,LocalDate 类确实是一个没有时间的日期,也没有区域/偏移量。您将看到不受Daylight Saving Time (DST) 影响的稳定日期值。

解析。

LocalDate ld = LocalDate.parse( "2019-01-23" ) ;

商店。

myPreparedStatement.setObject( … , ld ) ;

检索。

LocalDate ld = myResultSet.getObject( … , LocalDate.class ) ;

关于LocalDate

LocalDate 类表示仅日期值,没有时间,也没有 time zoneoffset-from-UTC

时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因区域而异。例如,Paris France 中午夜后几分钟是新的一天,而 Montréal Québec 中仍然是“昨天”。

如果没有指定时区,JVM 会隐式应用其当前的默认时区。该默认值可能在运行时(!)期间change at any moment,因此您的结果可能会有所不同。最好将您想要/预期的时区明确指定为参数。如果关键,请与您的用户确认该区域。

Continent/Region 的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland。切勿使用 2-4 个字母的缩写,例如 ESTIST,因为它们不是真正的时区,没有标准化,甚至不是唯一的 (!)。

ZoneId z = ZoneId.of( "America/Montreal" ) ;  
LocalDate today = LocalDate.now( z ) ;

如果你想使用 JVM 当前的默认时区,请求它并作为参数传递。如果省略,代码会变得难以阅读,因为我们不确定您是否打算使用默认值,或者您是否像许多程序员一样没有意识到这个问题。

ZoneId z = ZoneId.systemDefault() ;  // Get JVM’s current default time zone.

或者指定一个日期。您可以通过数字设置月份,1 月至 12 月的编号为 1-12。

LocalDate ld = LocalDate.of( 1986 , 2 , 23 ) ;  // Years use sane direct numbering (1986 means year 1986). Months use sane numbering, 1-12 for January-December.

或者,更好的是,使用预定义的Month 枚举对象,一年中的每个月一个。提示:在整个代码库中使用这些 Month 对象,而不是仅仅使用整数,以使您的代码更具自我记录性、确保有效值并提供 type-safetyYearYearMonth 同上。

LocalDate ld = LocalDate.of( 1986 , Month.FEBRUARY , 23 ) ;

关于java.time

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

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

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

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

从哪里获得 java.time 类?

ThreeTen-Extra 项目通过附加类扩展了 java.time。该项目是未来可能添加到 java.time 的试验场。您可以在这里找到一些有用的类,例如IntervalYearWeekYearQuartermore

【讨论】:

    猜你喜欢
    • 2011-08-14
    • 2019-05-26
    • 2013-09-22
    • 1970-01-01
    • 2018-01-01
    • 2012-07-03
    • 2017-09-15
    • 2020-09-12
    • 2020-03-01
    相关资源
    最近更新 更多