XMLGregorianCalendar 到 java.sql.Timestamp
-
13-12-2019 - |
题
我正在使用网络服务,插入记录,返回 XMLGregorianCalendar 类型的时间戳值。我需要将其转换为 java.sql.Timestamp 值,因此我使用这样的函数。
public static java.sql.Timestamp getSqlTimeStamp(XMLGregorianCalendar xgc) {
if (xgc == null) {
return null;
} else {
return new Timestamp(xgc.toGregorianCalendar().getTime().getTime());
}
}
Timestamp timestamp=getSqlTimeStamp(ExitInXMLGregor.getTimestamp());
我的问题是,在服务器中,插入记录时的时间戳值如下所示:10/03/2012 19:23:22
但是当我进行类型转换时,我获得的时间戳值如下:10/03/2012 17:23:22
服务器(Web服务所在的位置)中的时间比我的语言环境多2小时,并且由于某种原因,我在转换后获得了插入语言环境时间。问题是我确实需要获取服务器时间,因为在数据库中,时间戳值与服务器时间匹配,并且由于时间戳值不同,我在更新操作中遇到问题。
请,我将不胜感激任何形式的帮助。谢谢!
编辑:我找到了一个解决方案,但并不完全是我所需要的。当我将 java.sql.Timestamp 格式的时间戳转换为 XMLGregorian 时,我设置服务器的时区 (setTimeZone(TimeZone.getTimeZone("GMT+02:00")))。这实际上是有效的,但距离理想的解决方案还很远(时区甚至服务器可能会发生变化)如果能够动态地知道此时服务器的时区,那就太好了,但我不知道如何.. 。
public static XMLGregorianCalendar getXMLGregorianCalendar(Timestamp timestamp)
throws BaseException {
try {
GregorianCalendar gc = new GregorianCalendar();
gc.setTimeZone(TimeZone.getTimeZone("GMT+02:00"));
gc.setTimeInMillis(timestamp.getTime());
return DatatypeFactory.newInstance().newXMLGregorianCalendar(gc);
} catch (DatatypeConfigurationException ex) {
throw new BaseException(ex);
}
}
解决方案
我怀疑时间戳确实指定了正确的时间,只是没有正确的时区显示。 2012-10-03T17:23:22.342与2012-10-03T19:23:22.342 + 02:00,假设前者的(隐藏)时区是+00:00。
其他提示
太长了;博士
- 两个字符串代表同一时刻,只是简单地调整为时区,而没有注意到 与 UTC 的偏移量.
- 通过始终在此类字符串中包含偏移量和区域信息来避免此类混淆。
- 使用现代 java.time 取代麻烦的遗留类的类。(
Instant
, , 不是Timestamp
)
示例代码。
myPreparedStatement.setObject(
… ,
myXMLGregorianCalendar // If forced to work with a `javax.xml.datatype.XMLGregorianCalendar` object rather than a modern java.time class…
.toGregorianCalendar() // …convert to a `java.util.GregorianCalendar`, and then…
.toZonedDateTime() // …convert to modern `java.time.ZonedDateTime` class.
.toInstant() // Adjust to UTC by extracting an `Instant` object.
)
从 JDBC 4.2 及更高版本开始,从数据库检索。
Instant instant = myResultSet.getObject( … , Instant.class ) ;
java.time
作为 Robert Tupelo-Schneck 接受的正确答案 说,看起来一切都很好,因为两个字符串代表相同的时刻,但调整到不同的时区,与 UTC 的偏移量不同。问题是这两个字符串之一缺少其相对 UTC 偏移量的指示符。
这种遗漏是一种不好的做法,因为它会造成这种混乱。始终包含与 UTC 的偏移量,除非绝对确定上下文使偏移量/区域清晰。
在 UTC 工作
在 UTC 中工作可以避免这种混乱。通常最好以 UTC 格式工作、存储和交换日期时间值。从 UTC 调整为仅用于呈现给用户或业务逻辑需要的时区。
另外,您正在使用非常麻烦的旧类,这些类现在已被 java.time 类。转换你的 XMLGregorianCalendar
到 java.time.ZonedDateTime
.
ZonedDateTime zdt = myXMLGregorianCalendar.toGregorianCalendar().toZonedDateTime() ;
通过提取调整到 UTC Instant
目的。
Instant instant = zdt.toInstant() ;
从 JDBC 4.2 及更高版本开始,您可以直接交换 java.time 与数据库的对象。无需使用旧版 java.sql.Timestamp
又上课了。
myPreparedStatement.setObject( … , instant ) ;
恢复:
Instant instant = myResultSet.getObject( … , Instant.class ) ;
如果您想使用特定地区的人们使用的挂钟时间来查看同一时刻,请从 UTC 调整到某个特定时区。
ZoneId z = ZoneId.of( "Africa/Tunis" ) ;
ZonedDateTime zdt = instant.atZone( z ) ;
转变
如果您必须与一些需要 java.sql.Timestamp
, ,您可以来回转换 java.time.Instant
. 。调用添加到旧类中的新转换方法。
java.sql.Timestamp ts = java.sql.Timestamp.from( instant ) ;
往另一个方向走。
Instant instant = ts.toInstant() ;
也可以看看 我的答案 类似的问题。
关于 java.time
这 java.time 框架内置于 Java 8 及更高版本中。这些课程取代了麻烦的旧课程 遗产 日期时间类,例如 java.util.Date
, Calendar
, & SimpleDateFormat
.
这 乔达时间 项目,现在在 维护模式, ,建议迁移到 java.time 类。
要了解更多信息,请参阅 甲骨文教程. 。并在 Stack Overflow 上搜索许多示例和解释。规格为 JSR 310.
您可以交换 java.time 对象直接与您的数据库。用一个 JDBC驱动程序 符合 数据库连接4.2 或稍后。不需要字符串,不需要 java.sql.*
类。
从哪里获取 java.time 类?
- Java SE 8, Java SE 9, Java SE 10, , 然后
- 内置。
- 具有捆绑实现的标准 Java API 的一部分。
- Java 9 添加了一些小功能和修复。
- Java SE 6 和 Java SE 7
- 许多 java.time 功能都向后移植到 Java 6 和 7 三十后端口.
- 安卓
- 更高版本的 Android 捆绑实现 java.time 类。
- 对于早期的 Android (<26), 三十ABP 项目适应 三十后端口 (上文提到的)。看 如何使用 ThreeTenABP....
这 三十额外 项目通过附加类扩展了 java.time。该项目是 java.time 未来可能添加的内容的试验场。您可能会在这里找到一些有用的类,例如 Interval
, YearWeek
, YearQuarter
, , 和 更多的.