TimeZoneInfo.Local与TimeZoneInfo.FindSystemTimeZoneById

我一直在使用DateTimeTimeZoneInfo类,我用以下代码遇到了一个有趣的结果:

 var dstStart = new DateTime(2013, 3, 10, 2, 0, 0, DateTimeKind.Local); var result = TimeZoneInfo.Local.IsDaylightSavingTime(dstStart); 

结果是False 。 我实际上会认为它是True (DST从3月10日凌晨2点开始)

然后我尝试使用FindSystemTimeZoneById代替类似的代码:

 var myTimeZone = TimeZoneInfo.FindSystemTimeZoneById("Eastern Standard Time"); var result = myTimeZone.IsDaylightSavingTime(dstStart); 

结果令人惊讶地True

然后,我检查了这些对象是否代表相同的时区:

 myTimeZone.Id == TimeZoneInfo.Local.Id // returns True (Both are "Eastern Standard Time") 

我的问题是: 为什么这些结果不同,更重要的是我如何才能使它们相同?

我的电脑绝对是Eastern Standard Time

更多信息:

我重新计算了我的计算机时钟,并运行了一些测试来比较上述每种方法返回的TimeZoneInfo对象。 这是我的测试程序

 var timeZoneFromLookup = TimeZoneInfo.FindSystemTimeZoneById("Eastern Standard Time"); var dstStart = new DateTime(2013, 3, 10, 2, 0, 0, DateTimeKind.Local); // -- The following return true -- Console.WriteLine("Equal? {0}", TimeZoneInfo.Local.Equals(timeZoneFromLookup)); Console.WriteLine("Has Same Rules? {0}", TimeZoneInfo.Local.HasSameRules(timeZoneFromLookup)); Console.WriteLine("Same Id? {0}", TimeZoneInfo.Local.Id == timeZoneFromLookup.Id); Console.WriteLine("Same Base UTC Offset? {0}", TimeZoneInfo.Local.BaseUtcOffset == timeZoneFromLookup.BaseUtcOffset); Console.WriteLine("Same Daylight Name? {0}", TimeZoneInfo.Local.DaylightName == timeZoneFromLookup.DaylightName); Console.WriteLine("Same Display Name? {0}", TimeZoneInfo.Local.DisplayName == timeZoneFromLookup.DisplayName); Console.WriteLine("Same Standard Name? {0}", TimeZoneInfo.Local.StandardName == timeZoneFromLookup.StandardName); Console.WriteLine("Same Support For DST? {0}", TimeZoneInfo.Local.SupportsDaylightSavingTime == timeZoneFromLookup.SupportsDaylightSavingTime ); Console.WriteLine("Same result as to whether date/time is ambiguous? {0}", timeZoneFromLookup.IsAmbiguousTime(dstStart) == TimeZoneInfo.Local.IsAmbiguousTime(dstStart) ); // -- The following return false -- Console.WriteLine("Same utc offset result? {0}", timeZoneFromLookup.GetUtcOffset(dstStart) == TimeZoneInfo.Local.GetUtcOffset(dstStart) ); Console.WriteLine("Same Conversion to UTC? {0}", TimeZoneInfo.Local.GetUtcOffset(dstStart) == timeZoneFromLookup.GetUtcOffset(dstStart) ); Console.WriteLine("Same result as to whether date/time is invalid? {0}", timeZoneFromLookup.IsInvalidTime(dstStart) == TimeZoneInfo.Local.IsInvalidTime(dstStart) ); Console.WriteLine("Same result as to whether date/time is DST? {0}", timeZoneFromLookup.IsDaylightSavingTime(dstStart) == TimeZoneInfo.Local.IsDaylightSavingTime(dstStart) ); 

我做了一点反思,我相信不一致源于System.TimeZoneInfo+CachedData.GetCorrespondingKind(TimeZoneInfo timeZone)如何仅在timeZone == this.m_localTimeZone (即,参数相同的情况下System.TimeZoneInfo+CachedData.GetCorrespondingKind(TimeZoneInfo timeZone)返回DateTimeKind.Local时实例,因为TimeZoneInfo.Local属性基于)。

如果您从TimeZoneInfo.FindSystemTimeZoneById传递了另一个TimeZoneInfo实例,我希望它返回DateTimeKind.Unspecified

这将(可能在其他方面)影响System.TimeZoneInfo.IsDaylightSavingTime(DateTime dateTime) ,其中,在dateTime.Kind是本地的情况下,它在本质上TimeZoneInfo.Local和您的TimeZoneInfo实例之间进行转换,并将转换基于GetCorrespondingKind对源和目标时区说(在源和目标都是本地的情况下,转换返回原始日期时间)。

使用非本地TimeZoneInfo时的行为差异在MSDN文档中定义。

第一个结果为False的原因是因为您创建的DateTime对象在技术上是不明确的。 2013年3月10日凌晨2点在美国东部时间本地时区没有。

doc声明IsDaylightSavingTime()方法“受TimeZoneInfo对象表示的时区与dateTime参数的Kind属性之间的关系影响”。 备注部分中的表格提供了每种可能组合的说明。

通过显式调用FindSystemTimeZoneById指定时区时,“Local Kind”DateTime参数首先从Local转换为指定的时区。 在此步骤中,模糊时间被解析为合法值。

尝试将此添加到您的测试中:

 var dstStart = new DateTime(2013, 3, 10, 2, 0, 0, DateTimeKind.Local); dstStart = dstStart.ToUniversalTime(); dstStart = TimeZoneInfo.ConvertTime(dstStart, TimeZoneInfo.Utc, myTimeZone); 

dstStart的最终值变为‘3/10/2013 3:00:00 AM’ (假设您的机器仍在EST中)。 IsDaylightSavingTime()在本地种类参数上发生了同样的转换,这说明了它返回True

这里真正令人惊讶的是IsDaylightSavingTime()方法在任何一种情况下都不会引发ArgumentException 。 文档说它会在给出DateTime类型的无效DateTime参数时抛出exception,但显然不会发生这种情况。

编辑:

在查看了TimeZoneInfo类的源代码和注释之后,我得出结论, IsDaylightSavingTime()方法并不意味着抛出exception。 这是文档中的错误。