夏令时和 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);
}
我对臭名昭著的夏令时有疑问。
我有一个数据点列表,一年中的每个小时一个。为了检查是否所有数据点都存在,我创建了一个时间迭代器,如下所示:
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);
}