java 中的sql.date_SQL DATE中的时区vs java.sql.Date

博客探讨了JDBC在处理时区差异时的复杂性,包括PostgreSQL 8.3之前的错误以及OracleDB的时区转换。文章指出,虽然JDBC规范未明确规定时区处理,但大多数RDBMS和JDBC驱动程序会考虑时区并进行适当处理。以H2和SQLite为例,展示了不同的处理方式。总结了SQL Server时区数据的复杂性,并强调了缺乏国际时区标准的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

小编典典

在JDBC规范不问候时区定义的任何细节。尽管如此,我们大多数人都知道必须处理JDBC时区差异的痛苦。

最终,日期/时间数据库类型的时区处理归结为数据库服务器,JDBC驱动程序以及两者之间的所有内容。您甚至受JDBC驱动程序错误的支配。PostgreSQL修复了版本8.3中的错误,其中

传递给Calendar对象的Statement.getTime,.getDate和.getTimestamp方法使时区朝错误的方向旋转。

当您使用创建新日期时new Date(0)(假设您正在使用Oracle JavaSE

java.sql.Date,则会创建您的日期

使用给定的毫秒时间值。如果给定的毫秒值包含时间信息,则驱动程序会将时间分量设置为默认时区(运行应用程序的Java虚拟机的时区)中与GMT零相对应的时间。

因此,new Date(0)应使用GMT。

调用时ResultSet.getDate(int),您正在执行JDBC实现。JDBC规范没有规定JDBC实现应如何处理时区细节。因此,您无法实现。从Oracle

11g

oracle.sql.DATEJavaDoc看,Oracle

DB似乎没有存储时区信息,因此它执行自己的转换以将日期转换为java.sql.Date。我没有使用Oracle

DB的经验,但是我想JDBC的实现是使用服务器的本地JVM的时区设置来进行从oracle.sql.DATE到的转换java.sql.Date。

您提到了多个RDBMS实现可以正确处理时区,但SQLite除外。让我们看一下将日期值发送到JDBC驱动程序以及从JDBC驱动程序获取日期值时H2和SQLite的工作方式。

使用此SQLite JDBC驱动程序,RS.getDate(int)代码要简单得多。它只是java.sql.Date使用long存储在数据库中的日期值返回a

因此,我们看到H2 JDBC驱动程序在处理带日期的时区转换方面很聪明,而SQLite

JDBC驱动程序则不明智(并不是说这个决定不明智,它可能很适合SQLite设计决定)。如果您追查提到的其他RDBMS

JDBC驱动程序的源代码,则可能会发现大多数驱动程序都以与H2相似的方式接近日期和时区。

尽管JDBC规范没有详细说明时区处理,但是RDBMS和JDBC实现设计人员考虑时区并会正确处理时区是很有意义的。特别是如果他们希望自己的产品在全球舞台上可销售。这些设计师非常聪明,即使没有具体规范,大多数人也都能做到这一点,我对此并不感到惊讶。

我在Microsoft SQL Server博客中找到了使用SQL Server

2008中的时区数据,它说明了时区如何使事情变得复杂:

时区是一个复杂的区域,每个应用程序都需要解决如何处理时区数据的问题,以使程序更加用户友好。

不幸的是,当前没有时区名称和值的国际标准权威。每个系统都需要使用自己选择的系统,并且直到有国际标准之前,尝试让SQL

Server提供一个系统都是不可行的,最终将导致比解决的问题更多的问题。

2020-09-11

### 回答1: java.util.DateJava提供的日期和时间的类,它包含了日期和时间的信息。它可以表示从1970年1月1日00:00:00以来经过的毫秒数。由于java.util.Date是不可变的类,因此它的操作非常有限。 而java.sql.Datejava.util.Date的一个子类,它是专门用于操作数据库中日期类型数据的类。java.sql.Date继承了java.util.Date的大部分方法,但是它只保存年月日的信息,不包含具体的时分秒。java.sql.Date可以通过java.sql包提供的接口和类与数据库交互。 因为java.sql.Date是专为数据库而设计的,它有很多特定的用法。比如可以将java.sql.Date从数据库中读取出来,并用作Java程序中的日期类型。此外,它还可以将Java程序中的日期类型转换为数据库中的日期类型,方便进行数据库操作。 总的来说,java.util.DateJava中常用的日期和时间类型,而java.sql.Date是专为数据库设计的日期类型,在与数据库交互时非常有用。 ### 回答2: java.util.DateJava中的一个类,它表示一个特定的时间点,包括年、月、日、时、分、秒等信息。它用于表示一个精确到毫秒的时间点,并且它没有与时区相关的信息。在使用java.util.Date时,我们可以通过构造方法设置指定的年月日时分秒,也可以使用现有的时间点创建对象。 java.sql.Datejava.util.Date的一个子类,用于在Java程序和数据库之间传递日期数据。它继承了java.util.Date的基本功能,同时还提供了一些用于处理数据库日期的特定方法。在Java中,java.sql.Date可以通过使用java.util.Date进行实例化,或者通过调用java.sql.Date的构造方法指定年月日。 java.util.Datejava.sql.Date在功能上有着一些区别。首先,java.util.Date中包含的时间信息更加详细,可以表示年、月、日、时、分、秒、毫秒等。而java.sql.Date只包含年、月、日的信息,没有时分秒。其次,java.util.Date可以用于处理任意的时间点,而java.sql.Date只能用于处理日期,时间部分会被忽略。另外,由于java.sql.Date主要用于数据库交互,它具有特定的格式,可以与数据库中的日期字段进行交互。 在实际应用中,我们通常使用java.util.Date来表示一个具体的时间点,比如记录日志的时间、定时任务的执行时间等。而在与数据库交互时,我们可以使用java.sql.Date来传递、存储和检索日期数据。 ### 回答3: java.util.Datejava.sql.Date都是Java中用于表示日期和时间的类,但在使用上有一些区别。 java.util.DateJava中最基本的日期和时间类,它包含了日期和时间的信息,可以表示从1970年1月1日00:00:00开始的时间戳。它可以表示精确到毫秒级别的时间,但是在处理日期和时间的各种操作时,它的功能相对较弱,需要借助其他类库进行处理。在Java 8之后,java.util.Date已经被废弃,推荐使用新的日期和时间API(java.time包)。 java.sql.Datejava.util.Date的一个子类,它用于在Java和数据库之间传递日期。它的底层实现是基于long类型的时间戳,但它对应的数据库类型是SQL中的DATE类型,只包含日期部分,不包含具体的时间信息。它的使用场景通常是在与数据库交互的过程中,需要将日期信息保存到数据库或从数据库中获取日期信息。 在实际使用中,如果我们需要进行日期和时间的大部分操作,推荐使用新的日期和时间API(java.time包),例如LocalDate、LocalTime和LocalDateTime等类,它们提供了更多的功能和操作,而且更易于使用和理解。而对于与数据库交互时,如果数据库的数据类型是DATE类型,我们可以使用java.sql.Date来表示和传递日期信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值