我有一个场景,我希望将UTC日期转换为带有特定时区的ISO 8601字符串,以通过web api向下发送。建议的方法是使用TimeZoneInfo,如下所示:
var configuredTimeZone = TimeZoneInfo.FindSystemTimeZoneById(timeZoneString);
var localTime = DateTime.SpecifyKind(TimeZoneInfo.ConvertTimeFromUtc(utcTime, configuredTimeZone), DateTimeKind.Local);
var stringResult = localTime.ToString("o");
这在我的本地计算机上工作正常,但我遇到了一个非常奇怪的情况,当代码托管在Azure Web应用程序中时,ToString输出不同的字符串。本地我得到2017-02-20T00:00:00-06:00(这是我想要的,因为它包含我需要的时区信息),但是当在Azure中托管时,我得到2017-02-20T00:00:00+ 00:00。(这是UTC,不是我想要的)。由于我手动应用我想要的时区,我不确定为什么格式附加了错误的时区信息。有没有人遇到过这个?
答案 0 :(得分:4)
正如评论中所提到的,System.DateTime
无法存储时区。它所知道的只是“本地”或“UTC”。在Azure中托管,其“本地”时间是 UTC。
因此,您对TimeZoneInfo.ConvertTimeFromUtc(utcTime, configuredTimeZone)
的陈述会将UTC时间转换为您的时间(20日午夜),但因为DateTime
没有时区,所以它是{{ 1}}。您的DateTimeKind.Unspecified
只是告诉它它是“本地”类型,如果是Azure主机,则为UTC + 00:00。
如果时区很重要,您希望使用System.DateTimeOffset代替DateTime.SpecifyKind(..., DateTimeKind.Local)
。它专门用于保持和操纵(顾名思义)时区偏移。