我一直在想,这两个属性的工作原理究竟是什么。我知道第二个是通用的,基本上不涉及时区,但谁能详细解释一下它们是如何工作的,以及在什么情况下应该使用哪个?
当前回答
这是个好问题。我恢复它是为了更详细地说明. net如何处理不同的Kind值。正如@Jan Zich指出的那样,它实际上是一个非常重要的属性,并且根据您使用Now还是UtcNow设置不同。
在内部,日期存储为Ticks(与@Carl Camera的答案相反),这取决于您是否使用Now或UtcNow。
DateTime。UtcNow的行为与其他语言相似。它将tick设置为基于GMT的值。它还将Kind设置为Utc。
DateTime。现在将Ticks值更改为如果它是您在GMT时区中的一天时间。它还将Kind设置为Local。
如果你晚了6个小时(GMT-6),你将得到6小时前的GMT时间。net实际上忽略了Kind,并将这个时间视为6小时前的时间,尽管它应该是“现在”。如果您创建一个DateTime实例,然后更改您的时区并尝试使用它,则会破坏更多。
具有不同“Kind”值的DateTime实例不兼容。
让我们看一些代码……
DateTime utc = DateTime.UtcNow;
DateTime now = DateTime.Now;
Debug.Log (utc + " " + utc.Kind); // 05/20/2015 17:19:27 Utc
Debug.Log (now + " " + now.Kind); // 05/20/2015 10:19:27 Local
Debug.Log (utc.Ticks); // 635677391678617830
Debug.Log (now.Ticks); // 635677139678617840
now = now.AddHours(1);
TimeSpan diff = utc - now;
Debug.Log (diff); // 05:59:59.9999990
Debug.Log (utc < now); // false
Debug.Log (utc == now); // false
Debug.Log (utc > now); // true
Debug.Log (utc.ToUniversalTime() < now.ToUniversalTime()); // true
Debug.Log (utc.ToUniversalTime() == now.ToUniversalTime()); // false
Debug.Log (utc.ToUniversalTime() > now.ToUniversalTime()); // false
Debug.Log (utc.ToUniversalTime() - now.ToUniversalTime()); // -01:00:00.0000010
正如您在这里看到的,比较和数学函数不会自动转换为兼容时间。时间间隔应该是将近一个小时,但实际上是将近6个小时。"utc < now"应该是真的(我甚至添加了一个小时来确保),但仍然是假的。
你也可以看到“周围的工作”,这是简单地转换为世界时的任何地方,Kind是不一样的。
我对这个问题的直接回答与公认答案中关于何时使用每种语言的建议一致。您应该总是尝试使用Kind=Utc的DateTime对象,除非在i/o(显示和解析)期间。这意味着您几乎总是应该使用DateTime。UtcNow,除了你创建对象只是为了显示它,并立即丢弃它的情况。
其他回答
这是个好问题。我恢复它是为了更详细地说明. net如何处理不同的Kind值。正如@Jan Zich指出的那样,它实际上是一个非常重要的属性,并且根据您使用Now还是UtcNow设置不同。
在内部,日期存储为Ticks(与@Carl Camera的答案相反),这取决于您是否使用Now或UtcNow。
DateTime。UtcNow的行为与其他语言相似。它将tick设置为基于GMT的值。它还将Kind设置为Utc。
DateTime。现在将Ticks值更改为如果它是您在GMT时区中的一天时间。它还将Kind设置为Local。
如果你晚了6个小时(GMT-6),你将得到6小时前的GMT时间。net实际上忽略了Kind,并将这个时间视为6小时前的时间,尽管它应该是“现在”。如果您创建一个DateTime实例,然后更改您的时区并尝试使用它,则会破坏更多。
具有不同“Kind”值的DateTime实例不兼容。
让我们看一些代码……
DateTime utc = DateTime.UtcNow;
DateTime now = DateTime.Now;
Debug.Log (utc + " " + utc.Kind); // 05/20/2015 17:19:27 Utc
Debug.Log (now + " " + now.Kind); // 05/20/2015 10:19:27 Local
Debug.Log (utc.Ticks); // 635677391678617830
Debug.Log (now.Ticks); // 635677139678617840
now = now.AddHours(1);
TimeSpan diff = utc - now;
Debug.Log (diff); // 05:59:59.9999990
Debug.Log (utc < now); // false
Debug.Log (utc == now); // false
Debug.Log (utc > now); // true
Debug.Log (utc.ToUniversalTime() < now.ToUniversalTime()); // true
Debug.Log (utc.ToUniversalTime() == now.ToUniversalTime()); // false
Debug.Log (utc.ToUniversalTime() > now.ToUniversalTime()); // false
Debug.Log (utc.ToUniversalTime() - now.ToUniversalTime()); // -01:00:00.0000010
正如您在这里看到的,比较和数学函数不会自动转换为兼容时间。时间间隔应该是将近一个小时,但实际上是将近6个小时。"utc < now"应该是真的(我甚至添加了一个小时来确保),但仍然是假的。
你也可以看到“周围的工作”,这是简单地转换为世界时的任何地方,Kind是不一样的。
我对这个问题的直接回答与公认答案中关于何时使用每种语言的建议一致。您应该总是尝试使用Kind=Utc的DateTime对象,除非在i/o(显示和解析)期间。这意味着您几乎总是应该使用DateTime。UtcNow,除了你创建对象只是为了显示它,并立即丢弃它的情况。
只是对上述要点的一点补充:DateTime结构体还包含一个鲜为人知的名为Kind的字段(至少,我很长时间都不知道它)。它基本上只是一个标志,指示时间是本地时间还是UTC时间;它不指定本地时间与UTC的实际偏移量。除了表明构造结构的意图之外,它还影响了ToUniversalTime()和ToLocalTime()方法的工作方式。
DateTime不知道时区是什么。它总是假设你在当地时间。UtcNow只意味着“从时间中减去我的时区”。
如果您想使用支持时区的日期,请使用DateTimeOffset,它表示带有时区的日期/时间。这是我吃了不少苦头才明白的。
DateTime。UtcNow告诉你的日期和时间是协调世界时,也被称为格林威治标准时间时区——基本上就像如果你在英国伦敦,但不是在夏天。DateTime。现在给出在当前区域显示的日期和时间。
我推荐使用DateTime。现在,无论何时你向人们展示一个日期——这样他们就会对他们所看到的值感到满意——他们可以很容易地将其与他们在手表或时钟上看到的值进行比较。使用DateTime。UtcNow当您想要存储日期或在以后的计算中使用它们时(在客户机-服务器模型中),您的计算不会被位于不同时区的客户机与服务器或彼此之间的客户机所混淆。
这个问题的“简单”答案是:
DateTime。现在返回一个DateTime值,表示当前系统时间(无论系统运行在哪个时区)。DateTime。Kind属性将为DateTimeKind。当地的
DateTime。UtcNow返回一个DateTime值,表示当前的通用协调时间(即UTC),无论系统的时区如何,该值都是相同的。DateTime。Kind属性将为DateTimeKind。Utc
推荐文章
- c# .NET中的App.config是什么?如何使用它?
- String类中的什么方法只返回前N个字符?
- 我如何提高ASP。NET MVC应用程序性能?
- 无法解析类型为“Microsoft.AspNetCore.Http.IHttpContextAccessor”的服务
- 如何在没有任何错误或警告的情况下找到构建失败的原因
- Visual Studio弹出提示:“操作无法完成”
- 否ConcurrentList<T>在。net 4.0?
- 在c#中解析字符串为日期时间
- 如何减去日期/时间在JavaScript?
- 由Jon Skeet撰写的《Singleton》澄清
- 自定义数字格式字符串始终显示符号
- Post参数始终为空
- string.ToLower()和string.ToLowerInvariant()
- 将Date对象转换为日历对象
- 检查instance是否属于某个类型