莫斯科时间错误转换为UTC

时间:2013-08-08 10:41:45

标签: c# .net

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

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。如何解决?

P.S。操作系统 - Windows 7企业版。

2 个答案:

答案 0 :(得分: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永久性地...... .... / / p>

示例代码:

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

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

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

答案 1 :(得分:3)

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

修补程序:Update for 2011 calendar history in Windows operating systems

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