DateTimeOffset 解析和自定义时区

DateTimeOffset parse and custom time zone

我们正在将 XML DateTime 值解析为 DateTimeOffset 值。根据 W3C XSD documentation for DateTime,该类型可能有时区信息,也可能没有。

我们的要求是:

问题在于,当 XML 没有时区的 DateTime 被解析为 DateTimeOffset 时,它默认使用本地(系统)时区。似乎无法覆盖默认时区,也无法确定时区是在内部解析还是添加。

有什么方法可以指定 DateTimeOffset 解析使用的默认时区吗?
如果不是,如何识别在解析 DateTimeOffset 时是否自动解析或添加了时区?

不支持像设置当前文化一样设置 .NET 应用程序的时区,这对我来说似乎很奇怪。

因此,解决此问题的唯一方法似乎是首先将值解析为 DateTime 并检查 Kind 属性。如果 Kind 不是 Unspecified,再次将值解析为 DateTimeOffset:

/*
sample values:
- 2015-06-03T10:47:01
- 2015-06-03T07:47:01Z
- 2015-06-03T10:47:01+03:00
*/

DateTimeOffset dto;
var timeZone = TimeZoneInfo.FindSystemTimeZoneById(ConfigurationManager.AppSettings["DefaultTimeZone"]);
var dt = DateTime.Parse(value);

if (dt.Kind == DateTimeKind.Unspecified)
{
    dto = new DateTimeOffset(dt, timeZone.GetUtcOffset(dt));
}
else
{
    dto = DateTimeOffset.Parse(value);
}

最简单的方法是提前测试字符串以查看它是否包含偏移量。正则表达式对此很有效。

下面是一个函数,应该可以很好地处理您描述的情况:

static DateTimeOffset ParseAsDateTimeOffset(string s, TimeSpan defaultOffset)
{
    if (Regex.IsMatch(s, @"(Z|[+-]\d{2}:\d{2})$"))
        return DateTimeOffset.Parse(s, CultureInfo.InvariantCulture);

    var dt = DateTime.Parse(s, CultureInfo.InvariantCulture);
    return new DateTimeOffset(dt, defaultOffset);
}

您也可以考虑稍微改变一下,即提供默认的 时区,而不是默认的 偏移量。这是一个重要的区别,因为时区的偏移量会根据特定日期是否处于夏令时期间而改变。另请参阅 the timezone tag wiki 中的 "Time Zone != Offset"。

static DateTimeOffset ParseAsDateTimeOffset(string s, TimeZoneInfo defaultTimeZone)
{
    if (Regex.IsMatch(s, @"(Z|[+-]\d{2}:\d{2})$"))
        return DateTimeOffset.Parse(s, CultureInfo.InvariantCulture);

    var dt = DateTime.Parse(s, CultureInfo.InvariantCulture);
    return new DateTimeOffset(dt, defaultTimeZone.GetUtcOffset(dt));
}