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

时间:2014-10-31 15:19:53

标签: java date datetime

如果我在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

为什么日期打印在计算机的默认时区中,而不是TimeZone构建Calendar

5 个答案:

答案 0 :(得分:8)

Answer by Joshi是正确但短暂的。我会稍微扩展一下,并展示现代课程的替代方案。

Date没有时区

您的代码calendar.getTime()java.util.Date对象中提取java.util.Calendar个对象。

java.util.Date类有一个善意但令人困惑的功能,其toString方法在生成字符串时应用JVM的当前默认时区。这就产生了Date类在没有时区时具有时区的错觉。 Date表示时间线上的一个时刻,作为自UTC 1970年第一时刻(1970-01-01T00:00:00Z)的纪元参考日期以来的毫秒数。 (更令人困惑的是,Date类实际确实的时区内部有一个时区,但是无法访问且与此讨论无关。)

避免遗留日期时间类

旧的日期时间类现在已经遗留下来,取而代之的是java.time类。而是只使用java.time类。

Instant

在UTC中,相当于java.util.Date,使用Instant类。 Instant类代表UTC中时间轴上的一个时刻,分辨率为nanoseconds(小数部分最多九(9)位)。

Instant instant = Instant.now();  // Current moment in UTC.

ZonedDateTime

要通过特定区域wall-clock time的镜头查看同一时刻,请应用ZoneId获取ZonedDateTime

continent/region的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland。切勿使用诸如ESTIST之类的3-4字母缩写,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

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

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

关于java.time

java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendar和& SimpleDateFormat

现在位于Joda-Timemaintenance mode项目建议迁移到java.time类。

要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310

从哪里获取java.time类?

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

答案 1 :(得分:4)

因为Date对象没有时区作为其状态的一部分,您需要SimpleDateFormat格式化并在您所需的时区打印日期

答案 2 :(得分:1)

将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中完全重做的原因)。

答案 3 :(得分:0)

您必须在打印日期/时间之前应用时区

例如: TimeZone.setDefault(TimeZone.getTimeZone( “IST”));

(代码未经测试)

答案 4 :(得分:-1)

Calendar#getTime() - 返回一个Date对象,表示此Calendar的时间值(距离Epoch的毫秒偏移量。准确地说,java.util.Date中的值是自Unix纪元以来的毫秒数,发生在1970年1月1日午夜,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()));