java - 为什么 1971 年 11 月 1 日之前的日期要在 java.util.Date 上添加一个小时?

标签 java datetime

以下代码似乎演示了 java.util.Date 中的一个错误,如果本地时钟设置为 GMT 并启用 DST 调整并且时间在 1971 年 11 月 1 日之前,则会添加一个小时。我的第一个假设始终是我搞错了。任何人都可以看到出了什么问题(或者这真的是一个 Java 错误)吗? 1971 年 11 月 1 日有什么重要意义?

import java.text.SimpleDateFormat;
import java.util.Locale;
import java.util.TimeZone;

class JavaUtilDateBug
{
    private static void demo() throws Exception
    {
        // UK developers usually have the clock on their development machines set
        // to "Europe/London" (i.e. GMT with daylight saving). Set it explicitly 
        // here so readers in other countries can see the problem too.
        TimeZone.setDefault(TimeZone.getTimeZone("Europe/London"));
        Locale.setDefault(Locale.ENGLISH);

        SimpleDateFormat dateFormat = new SimpleDateFormat("EEE MMM dd HH:mm:ss z yyyy");
        String strJan1st1970Expected = "Thu Jan 01 00:00:00 GMT 1970";
        String strJan1st1970Actual = dateFormat.parse(strJan1st1970Expected).toString();
        System.out.println("strJan1st1970Actual: " + strJan1st1970Actual); // -> "Thu Jan 01 01:00:00 GMT 1970"
        boolean jvmHasDateBug = !strJan1st1970Expected.equals(strJan1st1970Actual);
        System.out.println("jvmHasDateBug: " + jvmHasDateBug); // -> true

        // The anomaly only seems to affect times before 1 Nov 1971.
        final String strNov1st1971 = "Mon Nov 01 00:00:00 GMT 1971";
        assert strNov1st1971.equals(dateFormat.parse(strNov1st1971).toString());
    }

    public static void main(String[] args)
    {
        try
        {
            demo();
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
    }
}

我的Java环境:

  java version "1.6.0_13"
  Java(TM) SE Runtime Environment (build 1.6.0_13-b03)
  Java HotSpot(TM) Client VM (build 11.3-b02, mixed mode, sharing)

最佳答案

1968 年 10 月 27 日至 1971 年 10 月 31 日期间对英国标准时间进行了一次试验,我怀疑这就是导致此问题的原因。

这里有一些试验细节:

http://en.wikipedia.org/wiki/British_Summer_Time#Single.2FDouble_Summer_Time

1970 年 1 月 1 日欧洲/伦敦的时区是英国标准时间 (GMT+1),因此当您使用 java.text.SimpleDateFormat 解析 Jan 01 00:00:00 GMT 1970 时它生成的正确纪元值等于 BST 中的 Jan 01 01:00:00 1970。

然后,由于 java.util.Date 的糟糕性,当您调用 java.util.Date.toString() 时,它会使用当前的默认时区截至现在,已更改为 GMT,您将得到 Jan 01 01:00:00 GMT 1970。

关于java - 为什么 1971 年 11 月 1 日之前的日期要在 java.util.Date 上添加一个小时?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15300517/

相关文章:

java - sybase中临时表提交事务

java - TableRow 权重缩放与我在 Android java 应用程序中使用 GraphView 时的预期相反

java - tomcat 和 Apache 之间的适配器

c++ - 使用 strptime/strftime 的结果不一致

C# 日期时间堆栈内存

java - 如何构建 REST dto?

java - 适用于未签名小程序的默认 Java 策略文件在哪里?

Java8 appendPattern 与 appendValue 方法定义的模式产生不同的结果

arrays - 使用 matlab 将 yyyy-mm-dd hh :mm:ss. ms 转换为 yyyy-mm-dd hh:mm:ss

php - 如何在 php/mysql 中以 12 小时时间格式使用 >= AND <= 获取数据?