如何正确地将日期转换为UTC,然后将其转换回来?

我正在努力将DateTime转换为UTC,这个概念以及所有我无法正确理解的东西。

当我得到一个日期时间字符串,比如说“7/10/2013”​​,我就是这样做的

Convert.ToDateTime("7/10/2013").ToUniversalTime(); 

这将在数​​据库中将其记录为“7/10/2013 4:00:00 AM ”。 服务器位于美国东海岸(-5)。 当然,在2013年7月期间,DST仍然被观察到,因此在此期间的偏移是-4,这样额外的4小时4:00:00 AM “记录为UTC。

当我写这篇文章时,它是2014年2月,DST没有生效,所以现在偏移是-5。 在我的应用程序中,这是我在我的应用程序中选择的偏移量。

如果我将-5偏移应用于“ 7/10 / 2013 4:00:00 AM ”,则日期将为“ 7/09/2013 11:00:00 PM ”。

哪一天是错的,一天就关闭了。

问题#1

那么如何才能正确转换UTC时间呢? 这意味着,当用户现在在2014年2月加载我的应用程序时(当前时区偏移-5),2013年7月10日凌晨4:00:00应该仍然是2013年7月10日,而不是7/09/2013。

令我困惑的是,因为.ToUniversalTime()考虑了服务器DST,是否存在一个硬盘“通用时间”,它不受服务器所在位置的影响?

问题2

当我在西海岸和东海岸都有服务器写入数据库时​​会发生什么? 如果记录的UTC时间是基于东海岸或西海岸,应用程序如何判断?

基本上,代码怎么说,“7/10/2013 4:00:00 AM”是在东海岸创建的UTC时间(表示美国东海岸的7/10/2013 00:00:00 AM)和不是西海岸的服务器(这表明它是美国西海岸的7/09/2013 20:00:00 pm)?

对不起,如果这听起来很愚蠢。 任何建议表示赞赏。

==========最终编辑,我目前的解决方案===============

MiMo的答案是有道理的。 我对两件事感到困惑。

  1. UTC时间存储在数据库中意味着什么?
  2. 服务器的时间与应用程序用户的关系是什么?

我的应用程序可以被来自不同时区的用户使用,有些用户与服务器位于同一时区,有些则不是。 有些旅行,所以即使它们与服务器位于同一时区,它们也可能一直落在不同的时区。 我的应用程序允许他们选择他们所在的时区,并适当地反映时间。

最初,我只是从数据库中获取UTC时间并从中减去用户的时区偏移量。 正如Mimo所说,这是错误的。 原因可以在上面的post中看到。

我最初的解决方案是立即获取服务器的时区偏移并使用它来加入/减去UTC,这也是错误的。 截至2013年7月10日,当时服务器的偏移量为-4。 目前,在2014年2月,服务器时区偏移为-5。 解决方案当然是使用.ToLocalTime()

在我深入研究Mimo关于如何使用TimeZone.ToLocalTime()的建议之前,我将采取以下措施来暂时解决问题。

  1. 从数据库中获取UTC日期并转换为.ToLocalTime,这是服务器显示的内容。 所以到服务器,7/10/2013 4:00:00 AM成为7/10/2013 12:00:00 AM。

  2. 获取服务器时区偏移量。 目前显示-5,因为它在美国东海岸。

  3. 获取用户的时区偏移量。 对于西海岸,用户现在选择-8。 对于东海岸,用户现在选择-5。

  4. 获取用户时区和服务器时区之间的差异。 西海岸是-3。 东海岸是0。

  5. 从7/10/2013 12:00:00 AM减去差异,因此西海岸的截止日期为7/09/2013 21:00:00 PM,东海岸的截止日期为7/10/2013 12:00:00 AM。

全对了。

非常感谢你们。 现在是时候深入了解TimeZone.ToLocalTime()并看看我是否可以减少步骤2-5。

您使用ToLocalTime()转换回本地时间。 它会看到日期/时间是在七月,所以使用DST,因此它将移动4小时而不是5。

如果您有连接到服务器的客户端(例如Web浏览器),您最终希望将日期/时间转换为客户端的本地时间,而不是服务器的本地时间。 要做到这一点,最好的方法是使用TimeZone.ToLocalTime() :向服务器发送客户端所在的时区,然后直接转换为该时区。

永远不要添加/减去小时数 – 始终浏览时区并使用TimeZoone.ToLocalTime()。 当涉及DST时,添加/减去小时将不起作用。

请注意,无法从浏览器中获取当前(本地)时区。 如果您的客户端是浏览器,您需要从某种配置中获取时区或让用户输入它。

另请注意,一旦您开始处理不同的时区,您不再只有日期 – 您始终必须处理完整的日期时间:如果您剥离或松开时间部分,则所有转换将不再起作用。

关于问题2:UTC时间是通用的,不是基于任何特定时区,因此一旦转换为UTC,您就不必再担心服务器的时区了。

如果要将数据库中的本地时间存储为UTC,则需要先将其转换为通用时间:

  DateTime dbDateTime = localDateTime.ToUniversalTime(); ... store dbDateTime in the database ... 

当您从数据库中读回它时,它的Kind属性将设置为Unspecified。 您需要将其Kind属性显式设置为UTC:

 dbDateTime = ... get from database, eg (DateTime) reader["SomeDateTimeColumn"] dbDateTime = DateTime.SpecifyKind(dbDateTime, DateTimeKind.Utc); 

如果您想将其转换为本地时间,您可以使用:

 DateTime localDateTime = dbDateTime.ToLocalTime(); 

我有同样的问题,我做了以下扩展方法。

  public const string UTC = "UTC"; public static DateTime Convert(this DateTime date, string fromZone, string toZone) { TimeZoneInfo to = TimeZoneInfo.FindSystemTimeZoneById(toZone); TimeZoneInfo from = TimeZoneInfo.FindSystemTimeZoneById(fromZone); return new DateTime(TimeZoneInfo.ConvertTime(date, from, to).Ticks, DateTimeKind.Unspecified); } public static bool IsDayLightSaving(this DateTime date, string zone) { TimeZoneInfo to = TimeZoneInfo.FindSystemTimeZoneById(zone); return to.IsDaylightSavingTime(date); } public static DateTime ConvertToUTC(this DateTime date, string fromZone) { return date.Convert(fromZone, DateTime_Extension.UTC); } public static DateTime ConvertToLocal(this DateTime date, string toZone) { return date.Convert(DateTime_Extension.UTC, toZone); } 

我发现的问题实际上是找到当前用户的时区。 由于这没有在Header中传递(你可以通过JavaScript获得)我决定要求用户在注册时选择他们的时区

希望这可以帮助

以当地时间存储日期吗? 如果您只关心日期部分,那么您可能希望在将其保存到数据库之前将解析的时间强制为Utc:

 DateTime utcDate = DateTime.SpecifyKind(Convert.ToDateTime("7/10/2013"),DateTimeKind.Utc); 

这样,当它保存到数据库时,它已经是UTC,不包括时间组件。

也许早上太早了我需要另外一杯咖啡,但是(至少显示)不是只使用TimeZoneInfo.ConvertTimeFromUtc并指定用户所需的TZ的解决方案? 或者转换时间转换器没有考虑转换回所需TZ的日期/时间? 更深入思考:即使它确实如此,是否有人知道它是否考虑到服务器已转移到DST但用户所需的TZ还没有的边缘情况?