日历以错误的时区返回日期

如果我在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/MontrealAfrica/CasablancaPacific/Auckland 。 切勿使用3-4字母缩写,例如ESTIST因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

 ZoneId z = ZoneId.of( "Asia/Kolkata" ); ZonedDateTime zdt = instant.atZone( z ); 

搜索Stack Overflow以获取有关此主题的许多其他问题和解答。 这个问题实际上与许多其他问题重复。


关于java.time

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

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

要了解更多信息,请参阅Oracle教程 。 并搜索Stack Overflow以获取许多示例和解释。 规范是JSR 310 。

从哪里获取java.time类?

  • Java SE 8SE 9及更高版本
    • 内置。
    • 带有捆绑实现的标准Java API的一部分。
    • Java 9增加了一些小function和修复。
  • Java SE 6SE 7
    • 许多java.timefunction都被反向移植到ThreeTen-Backport中的 Java 6和7。
  • Android的
    • ThreeTenABP项目特别适用于Android的ThreeTen-Backport (如上所述)。
    • 请参见如何使用ThreeTenABP ….

ThreeTen-Extra项目使用其他类扩展了java.time。 该项目是未来可能添加到java.time的试验场。 您可以在这里找到一些有用的课程,如IntervalYearWeekYearQuarter等。

将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()));