我目前正在尝试确保我们的旧后端可以支持根据用户当前时区(或更具体地说是偏移量(解析日期时间。 我们的服务器采用东部标准时间,我们的大部分日期时间都来自那里。 但是,对于位于其他时区的用户,在检索这些日期时间时需要转换为其时区(在本例中为偏移量(。 此外,来自用户的日期时间必须转换为东部标准时间,然后才能在服务器上持久化。 鉴于我们正在开发的前端是基于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
,它肯定会更干净。
编辑:我现在添加了一种方法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();
}
补充一点,第一种方法可以在没有customResolver
的情况下重写。
using System;
using NodaTime;
namespace qwerty
{
class Program
{
static void Main(string[] args)
{
var convertedInUTC = ConvertToUtcFromCustomTimeZone("America/Chihuahua", DateTime.Now);
Console.WriteLine(convertedInUTC);
}
private static DateTime ConvertToUtcFromCustomTimeZone(string timezone, DateTime datetime)
{
DateTimeZone zone = DateTimeZoneProviders.Tzdb[timezone];
var localtime = LocalDateTime.FromDateTime(datetime);
var zonedtime = localtime.InZoneLeniently(zone);
return zonedtime.ToInstant().InZone(zone).ToDateTimeUtc();
}
}
}