我遇到时区问题。现在我正在从客户端保存时区并将所有日期时间存储在 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 时区落后的问题。我还附上了屏幕截图。
我需要如何处理这种情况?
最佳答案
主要问题是你转换了错误的方向。您正在从 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/