TimeZoneInfo.Local与TimeZoneInfo.FindSystemTimeZoneById
我一直在使用DateTime
和TimeZoneInfo
类,我用以下代码遇到了一个有趣的结果:
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。 这是文档中的错误。