11

我在使用 JSON 序列化对象在客户端浏览器上显示正确日期时遇到问题。用户能够定义他们想要查看数据的时区。鉴于此,我将 UTC 日期转换为服务器上用户的时区。然后我想通过 JSON 将日期/时间(已经转换为其定义的时区)序列化到浏览器。

看起来很简单,但是我一直在使用的 JSON 序列化程序严重破坏了我的日期。服务器使用 UTC,客户端使用 Central (-6)。即使我将 DateTime.Kind 指定为未指定,日期也会被调整(-12 小时)。

不知何故,.NET 知道客户端浏览器所在的时区以及服务器所在的时区,即使我已经根据用户的全局设置调整了时间并设置了日期,它也会从我的日期/时间中否定 -6种类不详。如何让 JSON 序列化程序不尝试调整我的日期?

List<ErrorGridModel> models = Mapper.Map<ErrorCollection, List<ErrorGridModel>>(errors);
foreach (ErrorGridModel m in models)
{
    //convert UTC dates to user local dateTime - split out date vs. time for grouping & splitting columns
    DateTime dtLocal = TimeZoneInfo.ConvertTimeFromUtc(m.ErrorDate, this.AppContext.User.TimeZoneInfo);
    m.ErrorDate = new DateTime(dtLocal.Year, dtLocal.Month, dtLocal.Day, 0, 0, 0, DateTimeKind.Unspecified);
    m.ErrorTime = new DateTime(1900, 1, 1, dtLocal.Hour, dtLocal.Minute, dtLocal.Second, DateTimeKind.Unspecified);
}
IQueryable<ErrorGridModel> dataSource = models.AsQueryable();
return new ContentResult() { Content = JsonConvert.SerializeObject(dataSource.ToDataSourceResult(request), new JsonSerializerSettings() { DateFormatHandling = DateFormatHandling.MicrosoftDateFormat }), ContentType = "application/json" };
//return Json(dataSource.ToDataSourceResult(request));

ISO 日期似乎有效,但我无法使用它们,因为我有需要旧版 Microsoft 格式的 3rd 方控件……它会调整我的时区。

4

3 回答 3

4

当你试图控制偏移量时,不要依赖DateTimeKind.Unspecified. 它有一些通常被解释为Unspecified == Local. 让 Json.Net 专门编码正确的偏移量(无论 ISO 或 MS 格式)的唯一方法是传递它 aDateTimeOffset而不是 a DateTime

// Start with the UTC time, for example your m.ErrorDate.
// Here I demonstrate with UtcNow.  Any DateTime with .Kind = UTC is ok.
var dt = DateTime.UtcNow;

// Use the appropriate time zone, here I demonstrate with EST.
var tzi = TimeZoneInfo.FindSystemTimeZoneById("Eastern Standard Time");

// Get the offset from UTC for the time zone and date in question.
var offset = tzi.GetUtcOffset(dt);

// Get a DateTimeOffset for the date, and adjust it to the offset found above.
var dto = new DateTimeOffset(dt).ToOffset(offset);

// Serialize to json
var json = JsonConvert.SerializeObject(dto, new JsonSerializerSettings
    {
        DateFormatHandling = DateFormatHandling.MicrosoftDateFormat,
    });


// The json should now contain the correct time and offset information.
// For example,  "\/Date(1358789156229-0500)\/"

现在希望您会发现您正在使用的 javascript 控件将遵守偏移量并适当地应用它。如果不是,那么剩下的问题是特定于您正在使用的控件。

于 2013-01-21T17:29:31.077 回答
2

这是关于我所处的确切情况的长时间讨论。 http://www.telerik.com/community/forums/aspnet-mvc/grid/grids-and-dates.aspx

最重要的是,如果您使用 Microsoft JSON 日期格式,它将始终以 UTC 格式反映日期,即从 1970 年 1 月 1 日 UTC 开始的毫秒数(滴答声)。我无法在服务器上将时间自动转换为本地时间,并通过 JSON 将它应该下降的时间发送到 Kendo Grid,因为 Kendo Grid 控件将 JS 中的刻度中的日期实例化为 UTC。显示此日期时,它会自动将该值转换为浏览器的本地时区。

从服务器显示我的服务器转换后的日期值的唯一方法是通过 JSON 将日期作为字符串发送到客户端。

于 2013-01-22T01:27:52.087 回答
0

我们也遇到了这个问题。正如您所注意到的,问题实际上发生在客户端。通过在网格中使用请求结束处理程序,您可以将日期转换回 UTC。在这里找到的例子:

http://www.kendoui.c​​om/code-library/mvc/grid/using-utc-time-on-both-client-and-server-sides.aspx

于 2013-03-05T22:06:58.923 回答