6

如果当前位置是莫斯科,则下一个代码显示错误的当地时间:

DateTime dt = new DateTime(2010, 1, 1, 10, 0, 0, 0, DateTimeKind.Utc);
Console.WriteLine(dt + " - " +  dt.ToLocalTime());

dt = new DateTime(2010, 7, 1, 10, 0, 0, 0, DateTimeKind.Utc);
Console.WriteLine(dt + " - " + dt.ToLocalTime());

输出:

01.01.2010 10:00:00 - 01.01.2010 14:00:00
01.07.2010 10:00:00 - 01.07.2010 15:00:00

应该是 13:00 和 14:00。如何解决?

PS 操作系统 - Windows 7 企业版。

4

2 回答 2

7

看起来原因是当新的俄罗斯法律生效时,微软的家伙,而不是让莫斯科时间永久开启 DST,而是改变了它的基本 UTC 偏移量并禁用了 DST 切换。由于时区信息不包含任何 UTC 偏移的历史定义(与历史 DST 切换相反),这破坏了旧的日期时间转换。因此,结果如下:

  • 01/01/2011 10:00:00 = 01/01/2011 14:00:00 因为它是“GMT+4”
  • 01/07/2010 10:00:00 = 01/07/2010 15:00:00 因为它是“GMT+4 + DST”

这显然是 Microsoft 时间库中的一个错误。至少他们可以将其表示为 DST 永久开启....

示例代码:

void Main()
{
    DateTime dt = new DateTime(2010, 1, 1, 10, 0, 0, 0, DateTimeKind.Utc);
    UTCToLocal(dt);

    dt = new DateTime(2010, 7, 1, 10, 0, 0, 0, DateTimeKind.Utc);
    UTCToLocal(dt);
}

void UTCToLocal(DateTime datetime) {
    if (datetime.Kind != DateTimeKind.Utc)
        throw new ApplicationException();

    Console.WriteLine("UTC: {0} MOW: {1} IsDST: {2}", datetime, datetime.ToLocalTime(),TimeZoneInfo.Local.IsDaylightSavingTime(datetime.ToLocalTime()));
}

输出:

UTC: 01/01/2010 10:00:00 MOW: 01/01/2010 14:00:00 IsDST: False
UTC: 01/07/2010 10:00:00 MOW: 01/07/2010 15:00:00 IsDST: True

您可以清楚地看到夏季日期时间作为 DST 时间处理。

编辑:问题的解决方案在下一个答案中,我将把我的问题留在这里作为背景。

于 2013-08-08T11:25:23.763 回答
3

这是 Windows 历史时区数据中引入的错误。有一个可用的修补程序(这适用于 Windows 7、Server 2008 R2 和其他系统,而不仅仅是 XP)。

修补程序:Windows 操作系统中 2011 年日历历史记录的更新

正如 JMK 在评论中所说,您可能需要考虑使用优秀的库NodaTime,特别是如果您将程序分发给可能没有此修补程序的用户。

于 2013-08-08T11:41:51.680 回答