夏令时和 DateTime.AddHour()

Daylight saving and DateTime.AddHour()

我对臭名昭著的夏令时有疑问。

我有一个数据点列表,一年中的每个小时一个。为了检查是否所有数据点都存在,我创建了一个时间迭代器,如下所示:

DateTime timeIterator = new DateTime(year, 1, 1, 0, 0, 0);

迭代它
timeIterator = timeIterator.AddHours(1);

并在每次迭代时检查列表中是否存在该时间点。

问题出现在夏令时。我以 2014 年为例,时钟在 3 月 30 日从 0200 移动到 0300。因此,在 0159 之后是 0300。但是 DateTime.AddHours() 完全忽略了夏令时。如果 timeIterator 在 {30.03.2014 01:00:00} 并且我调用 AddHours(1) 我得到 {30.03.2014 02:00:00} 这显然不存在。

如果我现在根据这个数据点测试列表,它(自然地)不在列表中并且我抛出错误 "missing datapoint error"。

如何检查我的 DateTime 是否是有效的时间点?

提前致谢,
弗兰克

How can I check, if my DateTime is a valid point in time?

不检查,确保。例如,您可以使用 UTC 计算小时数并将每个点转换为本地时间。

DateTimeOffset timeIterator = new DateTimeOffset(new DateTime(year, 1, 1, 0, 0, 0, DateTimeKind.Local));
timeIterator = timeIterator.AddHours(1);
timeIterator.LocalDateTime;

// assuming you have a TimeZoneInfo object, you can also get different local times:
TimeZoneInfo tzi = /* the timezone */;
TimeZoneInfo.ConvertTimeFromUtc(timeIterator.UtcDateTime, tzi);

然而,正如Damien_The_Unbeliever评论的那样,这只解决了问题的一部分,程序将不再检查是否存在无效日期。当您的数据与当地时间一起存储时,会出现一个不同的问题。这意味着从 DST 到正常时间的转换将有两个连续的小时,具有相同的本地 DateTime 表示。为了避免这种情况,数据必须以完整的信息(UTC或显式偏移量)存储,以便以后进行比较。

编辑:

如果您的数据时区有一个有效的 TimeZoneInfo 对象,并且您出于某种原因不想切换到 DateTimeOffset,您还可以使用以下函数:

TimeZoneInfo tzi = /* the timezone */;
DateTime timeIterator = /* the time */;
if (tzi.IsAmbiguousTime(timeIterator))
{
    /* Expect multiple data entries */
}
if (tzi.IsInvalidTime(timeIterator))
{
    /* Expect no data entries */
}

我也有时区问题。 我建议您将日期保存为 UTC 值

然后改造一下:(可能使用扩展的方法)

private DateTime TransformToTimezone(DateTime datetime)
{
            TimeZoneInfo beZone = TimeZoneInfo.FindSystemTimeZoneById("Romance Standard Time");

            //Convert to timezone
            return TimeZoneInfo.ConvertTimeFromUtc(datetime, beZone);
}