日历以错误的时区返回日期
如果我在GMT时区的计算机上执行以下操作
TimeZone timeZone = TimeZone.getTimeZone('IST'); // India Standard Time Calendar calendar = Calendar.getInstance(timeZone); System.out.println(calendar.getTime());
它打印
10月31日星期五15:18:22 GMT 2014
为什么日期打印在计算机的默认时区,而不是Calendar
构建的TimeZone
?
因为Date
对象没有时区作为其状态的一部分,您需要SimpleDateFormat格式化并在您需要的时区打印日期
Joshi的回答是正确但简短的。 我会扩展一下,并展示现代课程的替代方案。
Date
没有时区
您的代码calendar.getTime()
从java.util.Calendar
对象中提取java.util.Calendar
对象。
java.util.Date
类有一个善意但令人困惑的特性,其toString
方法在生成字符串时应用JVM的当前默认时区。 这就产生了一种错觉,即Date
类没有时区。 Date
表示时间线上的时刻,作为自1970年UTC第一时刻( 1970-01-01T00:00:00Z
)的纪元参考日期以来的毫秒数。 (更令人困惑的是, Date
类实际上确实有一个深入内部的时区,但是无法访问且与此讨论无关。)
避免遗留日期时间类
旧的日期时间类现在是遗留的,取而代之的是java.time类。 而是只使用java.time类。
Instant
对于UTC中的一个时刻,相当于java.util.Date
,使用Instant
类。 Instant
类表示UTC时间轴上的一个时刻,分辨率为纳秒 (最多九(9)位小数)。
Instant instant = Instant.now(); // Current moment in UTC.
ZonedDateTime
要通过镜头查看特定区域的挂钟时间 ,请应用ZoneId
以获取ZonedDateTime
。
以continent/region
的格式指定适当的时区名称 ,例如America/Montreal
, Africa/Casablanca
或Pacific/Auckland
。 切勿使用3-4字母缩写,例如EST
或IST
因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
ZoneId z = ZoneId.of( "Asia/Kolkata" ); ZonedDateTime zdt = instant.atZone( z );
搜索Stack Overflow以获取有关此主题的许多其他问题和解答。 这个问题实际上与许多其他问题重复。
关于java.time
java.time框架内置于Java 8及更高版本中。 这些类取代了麻烦的旧遗留日期时间类,如java.util.Date
, Calendar
和SimpleDateFormat
。
现在处于维护模式的Joda-Time项目建议迁移到java.time类。
要了解更多信息,请参阅Oracle教程 。 并搜索Stack Overflow以获取许多示例和解释。 规范是JSR 310 。
从哪里获取java.time类?
- Java SE 8和SE 9及更高版本
- 内置。
- 带有捆绑实现的标准Java API的一部分。
- Java 9增加了一些小function和修复。
- Java SE 6和SE 7
- 许多java.timefunction都被反向移植到ThreeTen-Backport中的 Java 6和7。
- Android的
- ThreeTenABP项目特别适用于Android的ThreeTen-Backport (如上所述)。
- 请参见如何使用ThreeTenABP ….
ThreeTen-Extra项目使用其他类扩展了java.time。 该项目是未来可能添加到java.time的试验场。 您可以在这里找到一些有用的课程,如Interval
, YearWeek
, YearQuarter
等。
将Calendar对象转换为Date对象时,时区将丢失。 日历本身使用时区。 以下打印到指定的时区。
TimeZone timeZone = TimeZone.getTimeZone('IST'); // India Standard Time Calendar calendar = Calendar.getInstance(timeZone); System.out.println(calendar.get(Calendar.DAY_OF_MONTH) + "/" + (calendar.get(Calendar.MONTH)+1) + "/" + calendar.get(Calendar.YEAR) + " Time: " + calendar.get(Calendar.HOUR_OF_DAY) + ":" + calendar.get(Calendar.MINUTE)); System.out.println(calendar.getTime());
(注意不要像这样使用Calendar API,它非常混乱。使用日期格式化api)
Date api有很多问题(这就是它在java 8中完全重做的原因)。
你必须在打印日期/时间之前应用时区
示例:TimeZone.setDefault(TimeZone.getTimeZone(“IST”));
(代码未经测试)
Calendar#getTime() – 返回一个Date对象,表示此Calendar的时间值(距离Epoch的毫秒偏移量。准确地说,java.util.Date中的值是Unix纪元以来的毫秒数,发生在1月1日午夜1970年,UTC。和java.util.Date没有特定的时区。
另一方面,你也可以通过这种方式实现你想要做的事情
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MMM-dd HH:mm:ss z"); sdf.setTimeZone(TimeZone.getTimeZone("IST")); System.out.println(sdf.format(new Date()));