国内大多数开发者使用的电脑,都是使用的北京时间,日常开发的过程中其实并没有什么不便;不过,等遇到了阿里云等云服务器,系统默认使用的时间大多为 UTC 时间,这个时候,时区和时间的问题,就是不容忽视的大问题。
概念
首先明确一点,对于一个时刻,不管你用 UTC 时间还是 UTC+8 的时间来表示,本质上是一个时刻,就是一样的。我们处理日期和时间的目标,也是为了保证这个时刻不会因为时区的不同出现对不上的情况。
DateTime 与 DateTimeOffset
.NET 中表示时刻的数据类型有这两个(新出的 Date 和 Time 不作讨论),关于这两个数据类型,已经有同学写的很清楚了,阿里云很多服务器使用的时间为 UTC 时间,这个时候,如果使用 DateTime,是很难说清楚时区(Kind 只有 UTC、Local 还有未指定,不支持特定的某个时区),因此我们应当优先使用 DateTimeOffset。
TimeZoneInfo
用于跨时区的情况下,时区的信息是很重要的,.NET 中使用 TimeZoneInfo 这个类表示时区的信息。该类提供了一些静态方法,可以用于查找时区和创建时区等等。最早我是倾向于使用这些方法找到东八区的信息的,但是我发现诸如 ConvertTimeBySystemTimeZoneId 和 FindSystemTimeZoneById 的方法,都依赖于系统中的定义,不同的系统可能还不一样,自己定义是比较保险的,于是,我使用了 CreateCustomTimeZone 来新建一个时区。
Unix 时间戳是比较于 1970 年的 UTC 标准时间,因此在处理的过程中,DateTime 的时间表示应当将它转换为 UTC 时间,以下的代码,是使用 TimeZoneInfo 实现时间转换的,使用的是 DateTime 数据类型。如果改用 DateTimeOffset,这个类型对转换为 Unix 时间戳更加友好。
internal static class DateTimeExtension
{
private static readonly TimeZoneInfo gmt8 = TimeZoneInfo.CreateCustomTimeZone("GMT+8", TimeSpan.FromHours(8), "China Standard Time", "(UTC+8)China Standard Time");
public static long ToUnixTime(this DateTime datetime)
{
DateTime dateTimeUtc = datetime;
if (datetime.Kind != DateTimeKind.Utc)
{
dateTimeUtc = datetime.ToUniversalTime();
}
if (dateTimeUtc.ToUniversalTime() <= DateTime.UnixEpoch)
{
return 0;
}
return (long)(dateTimeUtc - DateTime.UnixEpoch).TotalMilliseconds;
}
public static DateTime ToDateTime(this long unixTimestamp)
{
DateTime time = DateTime.UnixEpoch.AddMilliseconds(unixTimestamp);
return TimeZoneInfo.ConvertTimeFromUtc(time, gmt8);
}
public static DateTime ToDateTime(this long unixTimestamp, int timezone)
{
DateTime time = DateTime.UnixEpoch.AddMilliseconds(unixTimestamp);
return time.AddHours(timezone);
}
}
复制代码
其实,只要时区是正确的,那么可以也可以使用网友提供的方法进行转换。
// Code from https://stackoverflow.com/questions/5615538/parse-a-date-string-into-a-certain-timezone-supporting-daylight-saving-time
public DateTimeOffset ParseDateExactForTimeZone(string dateTime, TimeZoneInfo timezone)
{
var parsedDateLocal = DateTimeOffset.ParseExact(dateTime, "yyyy-MM-dd HH:mm:ss", CultureInfo.InvariantCulture);
var tzOffset = timezone.GetUtcOffset(parsedDateLocal.DateTime);
var parsedDateTimeZone = new DateTimeOffset(parsedDateLocal.DateTime, tzOffset);
return parsedDateTimeZone;
}
复制代码
实践指南
处理日期与时间的过程中,如果加入了 TimeZoneInfo 的情况下会使得程序变得非常麻烦,特别是各种 TimeZone 的 Id 和名称,不同系统也不统一的情况下,容易出现各种各样的问题。我想的就是避免用它,说说我的处理原则吧。
日期时间不使用 DateTime 类,全部使用 DateTimeOffset 类型
系统的内部处理,全部使用 UTC 标准时间进行数据表示
对于字符串的转换为 DataTimeOffset 的情况,显式指定时区的小时偏移量
直接使用时间的加减,避免使用时区的信息转换导致的代码复杂度增加
【可选】如果不用考虑 2038 年的情况下,可以考虑 Unix 时间戳简化时间表示
直接贴上我现在使用的代码段,思路就是在强制给字符串表示的时间,加上 UTC 标准时区信息,然后再修正时差。
public static class DateTimeExtension
{
public static long? ParseUnixTimeMillisecondsWithTimeZone(string datetimeString, string format = "yyyyMMddHHmmss", int timezoneOffset = 8)
{
//注意这里非常关键的参数DateTimeStyles.AssumeUniversal,就是设定数据都是UTC的,不管是不是,都强行指定为UTC,然后再按照时区的信息调整为正确的时间。
//给定的数据是东八区时间,但是加上这个参数,实际上的时间就会提前了8个小时,因此需要在后面的数据中直接减去8个小时,如果是其他地区的时间,那么也是一样操作。
if (!DateTimeOffset.TryParseExact(datetimeString, format, CultureInfo.InvariantCulture, DateTimeStyles.AssumeUniversal, out DateTimeOffset time)) return null;
DateTimeOffset dateTimeUtcOffset = time.AddHours(-timezoneOffset);
return dateTimeUtcOffset.ToUnixTimeMilliseconds();
}
public static DateTimeOffset ToDateTime(this long unixTimestamp) => DateTimeOffset.FromUnixTimeMilliseconds(unixTimestamp);
}
复制代码
对于 ASP.NET CORE,JSON.NET 会自动处理符合 ISO8601 规范的日期格式,只要指定数据类型为 DateTimeOffset,就能够准确转换了。
参考
评论