繁体   English   中英

通过Jackson序列化java.sql.Date,同时考虑夏令时

[英]Serializing java.sql.Date through Jackson while accounting for Daylight Savings Time

我正在使用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 (少了一个小时,这是夏令时影响的EST时间的多少) )。

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

提前致谢!

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.Date java.util.Date的确有一天的时间,并且使用UTC。 更令人困惑的是, java.util.Date在其源代码中隐藏了一个时区,而没有getters和setters,因此似乎尚不影响equals之类的方法的行为。 因此,尽管java.sql.Date具有名称,并且具有仅保留日期的目的,但实际上确实将时间设置为UTC。 因此, java.sql.Date在调整一天中的时间方面有些st头,这很可能是您遇到的问题。 这些传统的日期时间类是一大堆燕麦片。 您永远不要使用它们。

引用java.sql.Date的JavaDoc

毫秒值左右的精简包装,可让JDBC将其标识为SQL DATE值。 毫秒值表示自格林威治标准时间1970年1月1日00:00:00.000起经过的毫秒数。

为了符合SQL DATE的定义,必须通过将与实例相关联的特定时区中的小时,分​​钟,秒和毫秒设置为零,来“标准化” java.sql.Date实例包装的毫秒值。 。

顺便说一句, java.sql.Timestamp同样糟糕。 它也很笨拙地从java.util.Date继承,添加了第二个小数秒(纳秒)。 也避免使用此类,现在用java.time.Instantjava.time.OffsetDateTime 同样, java.sql.Timejava.time.LocalTime替换。

java.time.LocalDate是一个真正的日期

通过采用JSR 310和JDBC 4.2,您可以使用现代业界领先的java.time类。 到目前为止,Jackson可能已更新为使用java.time 如果不是,请参阅此问题以获取指向用于处理Jackson中的java.time的数据类型模块的链接。

看起来您的输入字符串采用标准ISO 8601格式YYYY-MM-DD。 解析/生成表示日期时间值的字符串时,默认情况下, java.time类使用标准格式。 对于仅日期使用, LocalDate类实际上是没有日期且没有区域/偏移的日期。 您将看到稳定的日期值,而不受夏令时(DST)的影响

解析。

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

商店。

myPreparedStatement.setObject( … , ld ) ;

检索。

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

关于LocalDate

LocalDate类表示仅包含日期的值,没有日期, 时间段offset-from-UTC

时区对于确定日期至关重要。 在任何给定时刻,日期都会在全球范围内变化。 例如, 法国巴黎午夜过后几分钟是新的一天,而在魁北克蒙特利尔仍然是“昨天”。

如果未指定时区,则JVM隐式应用其当前的默认时区。 该默认值可能会在运行时(!)期间随时更改 ,因此您的结果可能会有所不同。 最好将您的期望/期望时区明确指定为参数。 如果紧急,请与您的用户确认区域。

Continent/Region的格式指定正确的时区名称 ,例如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。

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对象,而不是仅使用整数,可以使您的代码更具自文档性,确保有效值并提供类型安全 YearYearMonth

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

关于java.time

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

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

现在处于维护模式Joda-Time项目建议迁移到java.time类。

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

在哪里获取java.time类?

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

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM