在 Java 中,当我使用
Calendar.getInstance();
时,我会得到当前时区的 Calendar
对象。但 java.sql.Timestamp
通常存储在 UTC 时间而不是本地时间。那么如何从日历实例中获取 UTC 时间呢?
DateFormat df = DateFormat.getTimeInstance();
Calendar cal = Calendar.getInstance();
Timestamp time = new Timestamp(cal.getTimeInMillis());
// Prints local time
System.out.println(df.format(new Date(time.getTime())));
// Printe local time, but I want UTT Time
System.out.println("timestamp: " + time);
当您调用
Calendar.getTime()
时,这将为您提供一个没有相关时区的值 - 或者您可以将其视为 UTC 格式 - 对于日历代表的 instant。因此,如果(例如)日历时区的上午 9 点,则可能是世界标准时间下午 5 点。
现在,当您调用
java.util.Date
(无论是隐式还是显式)时,Timestamp
(我想 toString()
)将在系统默认时区中进行 格式化- 但不要让这欺骗您,让您认为时区是对象本身数据的一部分。
您需要非常清楚确切地您想要实现的目标 - 然后您可能会发现Joda Time可以让您在代码中比内置库更清楚地表达这一点。
您的代码已经完全按照您的要求进行操作。
Timestamp
(以及 Date
)没有时区信息,应始终包含 GMT 时间戳(这是 Calendar.getTimeInMillis()
返回的内容)。
您看到打印当地时间的原因是
DateFormat
工厂方法以及Timestamp.toString()
隐式使用系统时区。
Instant.now()
您正在使用麻烦的旧日期时间类,现在已被 java.time 类取代。无需使用
DateFormat
、Calendar
或 Timestamp
。
Calendar
进行交互,请使用添加到旧类中的新方法进行转换。提取过时的 java.util.Date
,然后转换为 Instant
。
Instant instant = myCal.getTime().toInstant() ;
获取当前时刻作为
Instant
。 Instant
类表示 UTC 时间线上的时刻,分辨率为 纳秒(最多九 (9) 位小数)。
Instant instant = Instant.now() ;
2017-02-17T03:29:15.725Z
使用 JDBC 4.2 及更高版本,您可以直接与数据库交换
Instant
对象。所以不需要过时的java.sql.Timestamp
。
myPreparedStatement.setObject( … , instant ) ;
和检索:
Instant instant = myResultSet.getObject( … , Instant.class ) ;
显然您需要自 1970-01-01T00:00:00Z
的
纪元参考日期以来的毫秒数。您可以从
Instant
中提取该数字。但请注意数据丢失,因为 Instant
的分辨率为纳秒,您将其截断为毫秒。
long millis = instant.toEpochMilli() ;
1487302155725
java.time 框架内置于 Java 8 及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如
java.util.Date
、Calendar
和SimpleDateFormat
。
Joda-Time 项目现在处于维护模式,建议迁移到 java.time 类。
要了解更多信息,请参阅Oracle 教程。并在 Stack Overflow 上搜索许多示例和解释。规格为JSR 310。
从哪里获取java.time类?
The ThreeTen-Extra 项目通过附加类扩展了 java.time。该项目是 java.time 未来可能添加的内容的试验场。您可能会在这里找到一些有用的类,例如
Interval
、YearWeek
、YearQuarter
和 more。
以下片段应该满足您的要求:
df.setTimeZone(TimeZone.getTimeZone("UTC"));
这对我有用(代码在 Kotlin 中)
val cal = Calendar.getInstance()
cal.setTimeZone(TimeZone.getTimeZone("UTC"));
cal.setTimeInMillis(cal.getTimeInMillis());
return DateFormat.format(yyyy-MM-dd'T'HH:mm:ss,
cal).toString();