使用Noda Time在时区之间进行转换

我目前正在尝试确保我们的传统后端可以根据用户的当前时区(或更具体地说是偏移量)支持解析日期时间。 我们的服务器在东部标准时间,我们的大部分日期时间都来自那里。 但是,对于位于其他时区的用户,在检索这些日期时间时需要转换到其时区(或者,在这种情况下为偏移)。 此外,来自用户的日期时间必须在服务器上持久化之前转换为东部标准时间。 鉴于我们正在开发的前端是基于Web的,我能够在几分钟内检索用户的偏移量,并将该值传递到标头内的服务层。 我看了Noda Time并认为它是一个很棒的API。 它确实迫使我在更精致的事情中考虑时间,但我仍然不能100%确定我已正确使用它。 以下是我为上述转换编写的方法。 我测试了它们,它们似乎工作。 鉴于上面的场景,这看起来像是对图书馆的正确使用吗? 我正确地考虑约会时间吗?

public static DateTime ConvertToUtcFromEasternTimeZone(DateTime easternDateTime) { NodaTime.DateTimeZone easternTimeZone = NodaTime.DateTimeZoneProviders.Tzdb.GetZoneOrNull("America/New_York"); ZoneLocalMappingResolver customResolver = Resolvers.CreateMappingResolver(Resolvers.ReturnLater, Resolvers.ReturnStartOfIntervalAfter); var easternLocalDateTime = LocalDateTime.FromDateTime(easternDateTime); var easternZonedDateTime = easternTimeZone.ResolveLocal(easternLocalDateTime, customResolver); return easternZonedDateTime.ToDateTimeUtc(); } public static DateTime ConvertToEasternTimeZoneFromUtc(DateTime utcDateTime) { NodaTime.DateTimeZone easternTimeZone = NodaTime.DateTimeZoneProviders.Tzdb.GetZoneOrNull("America/New_York"); NodaTime.DateTimeZone utcTimeZone = NodaTime.DateTimeZoneProviders.Tzdb.GetZoneOrNull("UTC"); ZoneLocalMappingResolver customResolver = Resolvers.CreateMappingResolver(Resolvers.ReturnLater, Resolvers.ReturnStartOfIntervalAfter); var utcLocal = LocalDateTime.FromDateTime(utcDateTime); var utcZonedDateTime = utcTimeZone.ResolveLocal(utcLocal, customResolver); var easternZonedDateTime = utcZonedDateTime.ToInstant().InZone(easternTimeZone); return easternZonedDateTime.ToDateTimeUnspecified(); } public static DateTime ConvertToUtc(DateTime dateTime, int offsetInMinutes) { LocalDateTime localDateTime = LocalDateTime.FromDateTime(dateTime); var convertedDateTime = localDateTime.PlusMinutes(offsetInMinutes).ToDateTimeUnspecified(); return convertedDateTime; } public static DateTime ConvertFromUtc(DateTime dateTime, int offsetInMinutes) { LocalDateTime localDateTime = LocalDateTime.FromDateTime(dateTime); var convertedDateTime = localDateTime.PlusMinutes(-offsetInMinutes).ToDateTimeUnspecified(); return convertedDateTime; } 

这里的想法是,当我在UTC时间和数据库中的时区之间进行解析时,时区很重要。 当我在客户端时间和UTC时间之间解析时,偏移很重要。

将来,我们可以坚持UTC时间,这将更容易。 目前,这种解决方案是一个止损。

我们的想法是,我们将从……

客户端 – > UTC +/-偏移 – > UTC – >东部时间 – >数据库

数据库 – >东部时间 – > UTC – > UTC +/-偏移 – >客户端

最终……

客户端 – > UTC +/-偏移 – > UTC – >数据库

数据库 – > UTC – > UTC +/-偏移 – >客户端

你的第一种方法看起来没问题,虽然我们不知道customResolver是什么。

你的第二种方法有点过时了。 我建议:

 public static DateTime ConvertToEasternTimeZoneFromUtc(DateTime utcDateTime) { var easternTimeZone = DateTimeZoneProviders.Tzdb["America/New_York"]; return Instant.FromDateTimeUtc(utcDateTime) .InZone(easternTimeZone) .ToDateTimeUnspecified(); } 

请注意,您不需要在每个方法调用中查找东部时区 – 只需:

 private static readonly DateTimeZone EasternTimeZone = DateTimeZoneProviders.Tzdb["America/New_York"]; 

……然后到处使用它。

你的第三种和第四种方法不是我认为的惯用方法 – 对于你应该使用的第三种方法:

 public static DateTime ConvertToUtc(DateTime dateTime, int offsetInMinutes) { var offset = Offset.FromMinutes(offsetInMinutes); var localDateTime = LocalDateTime.FromDateTime(dateTime); return new OffsetDateTime(localDateTime, offset).ToInstant() .ToDateTimeUtc(); } 

第四种方法似乎有点棘手,因为我们没有提供与OffsetDateTime转换相关的OffsetDateTime 。 您使用的代码可能没问题,但如果您可以使用OffsetDateTime ,它肯定会更OffsetDateTime

编辑:我现在添加了一个Instant方法,使第四种方法更清洁。 它将成为1.2.0的一部分,您可以使用:

 public static DateTime ConvertFromUtc(DateTime dateTime, int offsetInMinutes) { var offset = Offset.FromMinutes(offsetInMinutes); var instant = Instant.FromDateTimeUtc(dateTime); return instant.WithOffset(offset) .LocalDateTime .ToDateTimeUnspecified(); }