如何让JSON.NET将日期/时间序列化为ISO 8601?

osc*_*tin 13 json iso8601 json.net asp.net-web-api

我有一个Web API应用程序,它将JSON返回给可能没有使用Microsoft技术的消费者.当我的控制器返回一个DateTime属性为JSON的对象时,它以这种格式序列化日期:

2017-03-15T00:00:00-04:00

这让消费者有点头疼,因为他们期望它采用ISO 8601格式.一些研究告诉我,JSON.NET现在默认使用ISO 8601(我使用的是9.0.1).当我运行此代码时......

Clipboard.Copy(JsonConvert.SerializeObject(DateTime.Now));
Run Code Online (Sandbox Code Playgroud)

......我明白了:

2017-03-15T09:10:13.8105498-04:00
Run Code Online (Sandbox Code Playgroud)

维基百科在表达完整日期和时间时将这些显示为有效的ISO 8601格式:

2017-03-15T11:45:42+00:00
2017-03-15T11:45:42Z
20170315T114542Z
Run Code Online (Sandbox Code Playgroud)

但是,我上面得到的输出并不完全匹配.我想要使​​用格式化程序2017-03-15T11:45:42Z.

并且可能完全值得另外一个问题,在我的Web API配置中添加以下行似乎被忽略,因为它继续在上面最初显示的日期返回JSON:

config.Formatters.JsonFormatter.SerializerSettings.Converters.Add(new IsoDateTimeConverter());
Run Code Online (Sandbox Code Playgroud)

我假设一旦找出核心问题,Web API问题也可能得到解决.

Bri*_*ers 24

你得到的格式 ISO 8601格式(阅读维基百科上的时间和时区指示符部分),只是你的日期显然没有调整到UTC时间,所以你得到的时区偏移附加到日期而不是Z祖鲁时区指标.

IsoDateTimeConverter有,你可以使用自定义它的输出设置.您可以通过设置DateTimeStyles为自动将日期调整为UTC AdjustToUniversal.如果不需要,您还可以自定义输出格式以省略小数秒.默认情况下,转换器不会调整为UTC时间,并且包含与秒可用的精度小数一样多的小数.

试试这个:

IsoDateTimeConverter converter = new IsoDateTimeConverter
{
    DateTimeStyles = DateTimeStyles.AdjustToUniversal,
    DateTimeFormat = "yyyy'-'MM'-'dd'T'HH':'mm':'ssK"
};

config.Formatters.JsonFormatter.SerializerSettings.Converters.Add(converter);
Run Code Online (Sandbox Code Playgroud)

如果你的日期已经是UTC,但DateTimeKind它们的设置不Utc应该是(例如它Unspecified),那么理想情况下你应该修改你的代码,以便在序列化之前正确设置这个指标.但是,如果您不能(或不想)这样做,您可以通过更改转换器设置来处理它,以便始终Z以日期格式包含指示符(而不是使用K查看DateTimeKind日期的说明符))并删除AdjustToUniversal指令.

IsoDateTimeConverter converter = new IsoDateTimeConverter
{
    DateTimeFormat = "yyyy'-'MM'-'dd'T'HH':'mm':'ss'Z'"
};
Run Code Online (Sandbox Code Playgroud)