javascript - C# - Asp.Net MVC 中的时区问题

标签 javascript c# asp.net-mvc timezone

我遇到时区问题。现在我正在从客户端保存时区并将所有日期时间存储在 UTC 中。它工作正常,但当我尝试将 DateTime 转换为位于 CST、EST、EDT 等时区之后的 UTC 时,它显示错误的数据。

问题 - 假设我在美国东部夏令时间晚上 10 点执行任何任务,它会在凌晨 2 点(按照 UTC)保存在数据库中,但是当我尝试获取一天的数据并传递当前 UTC 日期时。

我的问题是,如果我尝试获取像午夜 11 点(从美国东部标准时间)到当前时间的数据,但我从 UTC 到 EST 的转换是错误的,因为 UTC 午夜 12 点是昨天的晚上 8 点(因为比美国东部时间晚 4 小时)世界标准时间)。 (从日期[UTC 转换为 EDT] - 06/07/2017 08:00pm)到日期 - 06/07/2017 11:00 pm)由于这种转换,我只能从晚上 8 点到晚上 11 点获取数据 <-根据 UTC,我预计日期为 06/07/2017 04:00 AM。

代码- 下面是转换的代码。在起始日期中,我仅使用了 utcnow.date 并从日期开始

Javascript 代码 -

function setTimezoneCookie() {
            try {
                var timezone_cookie = "timezoneoffset";
                var timeZoneName = "timezonename"

                var tz = jstz.determine();
                var aa = tz.name();
                // if the timezone cookie not exists create one.

                if (!$.cookie(timezone_cookie)) {
                    // create a new cookie
                    $.cookie(timezone_cookie, new Date().getTimezoneOffset());
                    $.cookie(timeZoneName, aa);
                }
                else {
                    var storedOffset = parseInt($.cookie(timezone_cookie));
                    var currentOffset = new Date().getTimezoneOffset();
                    if (storedOffset !== currentOffset) {
                        $.cookie(timezone_cookie, new Date().getTimezoneOffset());
                        $.cookie(timeZoneName, aa);
                        location.reload();
                    }
                    else {
                        $.cookie(timeZoneName, aa);
                    }
                }
            }

c# 代码 -

 fromDate =Convert.ToDateTime(fromDate).ToClientTimeZoneinDateTime().ToString();  
                toDate = Convert.ToDateTime(toDate).ToClientTimeZoneinDateTime().ToString();

                ObjectParameter totalRecords = new ObjectParameter("TotalRecords", typeof(int));
                var DetailsList = objDetailsList.GetDetails(loginUserId,locationId, userId, taskType, pageIndex, numberOfRows, sortColumnName, sortOrderBy, textSearch, totalRecords, fromDate, toDate);
                if (DetailsList.Count() > 0)
                {
                    string output = BuildJQGridResults(DetailsList, numberOfRows, pageIndex, Convert.ToInt32(totalRecords.Value));
                    response.Write(output);
                }
                else
                {
                    JQGridResults result = new JQGridResults();
                    List<JQGridRow> rows = new List<JQGridRow>();
                    result.rows = rows.ToArray();
                    result.page = 0;
                    result.total = 0;
                    result.records = 0;
                    response.Write(new JavaScriptSerializer().Serialize(result));
                }

以下是将UTC时间转换为客户端时区的方法

public static DateTime ToClientTimeZoneinDateTime(this DateTime dt)
{
    try {               
        if (System.Web.HttpContext.Current.Request.Cookies["timezoneoffset"] != null || System.Web.HttpContext.Current.Request.Cookies["timezonename"] != null)
        {                  
            var timezonename = System.Web.HttpContext.Current.Request.Cookies["timezonename"].Value;
            timezonename = timezonename.Replace("%2F", "/");
            var timezoneLocal1 = FindTimezoneName(timezonename);
            TimeZoneInfo tzi = TimeZoneInfo.FindSystemTimeZoneById(timezoneLocal1);
            bool isCurrentlyDaylightSavings = tzi.IsDaylightSavingTime(dt);
            if (isCurrentlyDaylightSavings == true)
                dt.AddHours(1);

            var timeOffSet = System.Web.HttpContext.Current.Request.Cookies["timezoneoffset"].Value;

            var offset = int.Parse(timeOffSet.ToString());
            dt = dt.AddMinutes(-1 * offset);                  
            return dt;
        }             
        return dt.ToLocalTime();
    }
    catch (Exception)
    {
        return DateTime.UtcNow;
    }
}

毫无疑问,时区处理得当,但如果最终用户尝试在美国东部夏令时间晚上 8 点之后获取数据,则会面临 UTC 时区落后的问题。我还附上了屏幕截图。

下面是转换前的图片 - enter image description here

转换后的上图 - enter image description here

我需要如何处理这种情况?

最佳答案

主要问题是你转换了错误的方向。您正在从 UTC 转换为用户的时区,但您的输入位于用户的时区,因此您需要转换另一个方向 - 从用户的时区转换为 UTC。那么您的查询将显示更好的结果。

其他一些事情:

  • 不要通过尝试手动添加/减去分钟或小时来转换时区。使用 TimeZoneInfo 上提供的转换函数,如ConvertTimeFromUtc , ConvertTimeToUtc等。无需测试 DST。

  • try/catch 根本不应该出现在您的代码中。如果无法执行该操作,则抛出异常。不要通过吞咽异常来掩盖重要错误。

  • dt.ToLocalTime()也不应该出现在您的代码中。切勿依赖服务器的本地时区。

  • new Date().getTimezoneOffset() 返回的偏移量是用户的当前偏移量。您不能假设它是所选日期的正确偏移量。无论如何,您不需要这样做,因为您已经获得了时区名称。 (您根本不需要 timezoneoffset cookie。)

  • jstz.determine() 返回的时区名称客户端上的将是 IANA tzdb 标识符,例如 America/Los_Angeles 。这些不会在服务器端使用 TimeZoneInfo.FindSystemTimeZoneById 工作。 (除非您在 Linux 或 Mac 上运行 .NET Core)。需要转换为 Windows 时区。我看到你有一个FindTimeZoneName函数,我假设它正在执行转换。您没有在代码中显示详细信息,但我强烈建议您使用我的 TimeZoneConverter库来实现这一点,因为它是通过时区更改来维护的。

  • 读取 cookie 和时区转换是不同的问题。不要将它们捆绑在一起。

最终,你应该得到这样的结果:

public static DateTime FromTimeZoneToUtc(this DateTime dt, string timeZone)
{
    var windowsId = TimeZoneConverter.TZConvert.IanaToWindows(timeZone);
    var tzi = TimeZoneInfo.FindSystemTimeZoneById(windowsId);
    return TimeZoneInfo.ConvertTimeFromUtc(dt, tzi);
}

或者,如果您使用 Noda Time 则更好,那么您根本不需要转换时区。

public static DateTime FromTimeZoneToUtc(this DateTime dt, string timeZone)
{
    var tz = DateTimeZoneProviders.Tzdb[timeZone];
    var local = LocalDateTime.FromDateTime(dt);
    return local.InZoneLeniently(tz).ToDateTimeUtc();
}

关于javascript - C# - Asp.Net MVC 中的时区问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44439159/

相关文章:

c# - 在其父级边界之外的用户控件内显示控件

asp.net-mvc - 如何让过滤器重定向到另一个操作?

asp.net - Asp.Net Razor 的 ResolveUrl/ResolveClientUrl 等价物?

javascript - 对 jQuery 数据表中所有选定的行求和

javascript - 如何注入(inject)服务来监控身份验证和存储状态?

javascript - 使用reactjs获取API并显示在图表中

javascript - 有没有办法在 jquery 中没有 id 的 img 标签上设置背景图片?

C# XNA 桨球游戏

c# - 控件不在焦点时下拉列表更改所选值

asp.net - ASP.NET Forms 身份验证的授权安全