你需要了解的有关.NET日期时间的必要信息

640?wx_fmt=gif引言

    DateTime数据类型是一个复杂的问题,复杂到足以让你在编写【将日期从Web服务器返回到浏览器】简单代码时感到困惑。

ASP.NET MVC 5和 Web API 2/ASP.NETCore 以不同方式序列化日期,这可能会给在一个Web应用程序中同时使用这两个序列化的开发人员带来更多混淆。

本文会尽量覆盖 ASP.NET / ASP.NETCore 中与 Date/Time有关的歧义、参数绑定、序列化相关的知识点和坑位。 

Date/Time值歧义和应对策略

  使用Datetime最大的问题在于歧义,整个地球分为24个时区,每个时区都有自己的本地时间

例如一个DateTime值指示了本地时间,这个值在同时区的其他系统依然标示了同一时间点; 但是在其他时区以外,这个DateTime值就会有各自的解释。

电邮中若有

Sun , 04, June 2019 9:45:29 +0800

说明信件发送第时间是2019年6月04日,星期日,上午9点45分29秒,该地区领先UTC8小时 (+800,就是东八区时间)。

对于发这封邮件的时间,北京本地时9点45,伦敦是凌晨1点45 ......

  正因为这个现状,在Internet以及无线电通信时,时间的统一很重要。

  业界提出了不同的格式来明确地表示单一时间点, Web开发行业提出了不同的解决方案以明确定义时间值。

640?wx_fmt=gif  UTC时间

  在国际无线电通信中,为避免各自为战而普遍使用一个标准时间,称为协调世界时(UTC)

 UTC是0时区的时间值,与格林威治标准时间(GMT)一样,都与英国伦敦的当地时间相同

  UTC能准确代表单一时间点,UTC 时间对于住在加利福尼亚和中国的人来说都是一样的。

 如果把以上北京时间2019/06/04 09:45:29转化为UTC时间,可以使用以下公式: UTC + 时区差 = 当地时间。

因为北京时间是东8区,所以这个点UTC时间是 2019/06/04 02:45:29(也可认为该值是伦敦时间)

开发者可以考虑的解决方案 是: 使用UTC时间存储date/time、服务端维护、计算也使用UT时间、将UTC时间发送到浏览器,浏览器javascript在Web页上转化为当地时间。

640?wx_fmt=gif ISO 8601

  Web上有许多时间显示格式,但最著名并大规模采用的是ISO-8601 标准。https://www.cl.cam.ac.uk/~mgk25/iso-time.html

当没有更多信息的时候,只包含Date/Time 的写法被假定为当地时间,要指示该时间是UTC时间,可在Datetime值后面加上字母 z

在Datetime值后面增加 +hh:mm、 +hhmm, -hh 可指示该时间值相对于UTC时间的偏移

640?wx_fmt=jpeg

.Net中关于Date/Time的实现

  .Net 4.0+ 提供的各种结构已经全方位支持 date, timezone, timezone之间的转化,足以解决开发者遇到的Date/Time相关的问题。

DateTime

  DateTime 定义了一个特殊的date/time, 内置的Kind属性提供了受限的时区信息

  ① DateTimeKind.UTC 指定了UTC的DateTime,明确定义了单一时间点

  ② DateTimeKind.Local 指示了本地时间,这个值在具有相同时区的其他系统依然能够定义一个时间点,但是在其他时区以外,这个DateTime值会有不同解释。

  ③ DateTimeKind.Unspecified 更没有兼容性,仅表示时间值

  我们关注 DateTime.ToUniversalTime() 方法的表现,当DateTime被设定为Unspecified时候, ToUniversalTime会首先假定该值是 Local


640?wx_fmt=png


 when to use datetime?

  • 你只处理当地时间,你没有跨越时区的计算

  • 你只处理 UTC时间

  • 处理抽象日期/时间(使用 Unspecified):例如跨国公司的跨时区门店都在早上9点开业

DateTimeOffset

表示时间点,通常表示为一天中相对于UTC时间的日期/时间,该结构体自然带有相对于UTC时间的偏移信息

when to use DateTimeOffset?

  • 代码需要应对不同时区的时间值

  • 时区之间相互转化

  • 需要进行跨时区的计算

Date/Time 序列化

  Web API2 和ASP.NET Core内置的JSON序列化器是Newtonsoft.Json,JSON.NET将日期时间序列化为ISO-8601格式:

Date value

serialized value

DateTime.Now (Pacific time)

2016-10-11T19:25:34.8346658-07:00

DateTime.UtcNow

2016-10-12T01:25:34.8346658Z

new DateTime(2016, 6, 6, 4, 5,5 )

2016-06-06T04:05:05

new DateTimeOffset(new DateTime(2016, 6, 6, 4, 5,5 ), new TimeSpan(-2, 0,0))

2016-06-06T04:05:05-02:00

  ASP.NET MVC 5内置的

JSON Serializer 

还是System.Web.Script.Serialization.JavaScriptSerializer, 该序列化器将date序列化为时间戳:“\/Date(ticks)\/”, ticks 表示从1970-1-1 00:00:00 UTC(Unix Epoch)经历的毫秒数,

这样的格式在客户端需要使用JavaScript做一些本地转化, 转化后的值无法体现时区。 

640?wx_fmt=png


JavaScriptSerializer还有更多缺点,现在社区鼓励使用Json.Net 序列化器。

Date/Time字符串转换

  在开发中,常涉及Date/Time 参数绑定和字符串转换, 当中也有一些坑位需要规避。

一个时间日期字符串, 若没有相对于UTC的偏移信息,转换后的DateTime对象的DateTimKind是Unspecified;

若指定了offset,转换后的DateTime对象的DateTimeKind是Local, 并且时间值被调整到机器的当地时间。

最近我们生产环境WebSite再迁移到k8s集群 (UTC时间)之后,就遇到这样的问题:

订单转储的预期是:北京时间2019-05-11--->2019-05-12

实际情况是在网站端转换为Unspecified, 而进一步ToUniversalTime()过滤的时候,该时间段又被假定是机器的当地时间, 也就是说查询时间段变成了: 伦敦时间2019-05-11--->2019-05-12

这样自然与预期不符。

解决思路: 添加偏移信息,告知明确的时间段: 2019-05-11 00:00:00+08:00 ----> 2019-05-12 00:00:00+08:00

开发者每天都在使用的Date/Time有很多学问, 涉及Culture、Calendar、夏令时, 这里只是浅谈最常用的知识点和坑位。

原文地址:https://www.cnblogs.com/JulianHuang/p/10967289.html

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com 
640?wx_fmt=jpeg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/315659.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CF750F. New Year and Finding Roots

CF750F. New Year and Finding Roots Solution 设当前节点为xxx, 若当前节点为叶子节点,则向它的邻居跑。 若当前节点为根,则找到答案。 否则,每次从xxx向两个方向找,每次只走一个没有走过的邻居,显然可以…

只需3步,即可将你的Chromium Edge 浏览器设置成中文

最近,Chromium Edge 浏览器推出了添加语言包的功能,于是我们可以轻松将其界面设置成中文的。第1步: 升级浏览器到最新版在Chromium Edge 浏览器的地址栏中输入:edge://settings/help稍等片刻,浏览器会自行升级到最新版。第2步: 将Language se…

CF730F. Ber Patio

CF730F. Ber Patio Solution 由于∑ai\sum{a_i}∑ai​只有10510^5105,即除掉原有的bbb,代金券最多为10410^4104,因此我们令f[i][j]f[i][j]f[i][j]表示到了第iii天,靠现金获得的代金券共为jjj的最小现金和。 转移时枚举第i1i1i1…

【北京】线下活动 | Azure SQL Database Managed Instance发布会

6/12/2019 | 8.30 AM - 1.30 PM微软大厦一号楼一层故宫会议室N丹棱街5号 海淀区 北京市S尊敬的客户感谢您一直以来对微软的支持!诚挚地邀请您参加微软Azure SQL Database Managed Instance (SQL Database托管实例)在中国北京的发布会。SQL Da…

CF768F. Barrels and boxes

CF768F. Barrels and boxes Solution 又一道据说是套路题。 核心在于枚举WWW的连续段个数为xxx。 然后就能知道FFF的连续段个数为x−1,x,x1x-1,x,x1x−1,x,x1,可以用插板法求出任意把WWW分xxx段的方案数,分xxx段每段WWW至少HHH个的方案数,…

ASP.Net Core MVC 发生二次请求

Bug回忆录昨天搭建新框架的时候&#xff0c;遇到一个很奇怪的“Bug”&#xff0c;每次请求都会触发两次Aciton&#xff0c;举例子吧&#xff0c;Demo&#xff1a;_Layout.cshtmlHomeControllerIndex.cshtml<h1>Hi, Im index page.</h1>最简单不过的代码&#xff0c…

CF1354F. Summoning Minions

CF1354F. Summoning Minions Solution VPVPVP结束十分钟后AC,qwqAC,qwqAC,qwq。 首先因为ai,bi≥0a_i,b_i \geq 0ai​,bi​≥0&#xff0c;所以所有东西全用一次答案不会变劣&#xff0c;留到最后的集合大小一定为kkk。 如果我们知道留到最后的集合Sx1,x2...xkS{x_1,x_2...x…

DDD战略设计相关核心概念的理解

前言本文想再讨论一下关于领域、业务、业务模型、解决方案、BC、领域模型、微服务这些概念的含义和关系。初衷是我发现现在DDD领域建模以及解决方案落地过程中&#xff0c;常常对这些概念理解不清楚或者有歧义&#xff0c;导致我们不知道如何运用这些概念来落地我们的软件。领域…

CF1037H. Security

CF1037H. Security Solution 1 设原串为ststst。 对于单个询问&#xff0c;答案必然是询问串sss的一个前缀s[1..i]s[1..i]s[1..i]加上一个大于s[i1]s[i1]s[i1]的字符ccc构成。 因此我们只需要枚举前缀s[1..i]s[1..i]s[1..i]&#xff0c;枚举字符ccc&#xff0c;快速询问s[1…

Visual Studio容器项目工程化心得

引言  关注博主的网友会看到我使用ASP.NET Core 容器化部署企业级项目的过程&#xff0c; 回想到开发过程中&#xff0c;鄙人有一些工程化心得&#xff0c; 分享给同学们。项目工程化因为本项目涉及单元测试Project、容器化部署、本地化的sqlite数据库&#xff0c;nlog日志&a…

CF1223F. Stack Exterminable Arrays

CF1223F. Stack Exterminable Arrays Solution 奇怪的套路增加了。 大概就是[l,r][l,r][l,r]能匹配完可以转化为[1,l−1][1,l-1][1,l−1]匹配后的状态和[1,r][1,r][1,r]匹配后的状态相同&#xff0c;因此hashhashhash判断即可。 Code #include <vector> #include &l…

从严治码-系统集成项目之根本

从4月份开始&#xff0c;由于备考《系统集成项目管理工程师》的原因&#xff0c;博客没有持续更新&#xff0c;在上半年考试结束之后&#xff0c;又对项目进行了一些收尾的工作。下面就这段时间的学习作一个记录和总结吧。在学习的过程中&#xff0c;提炼了一些自己认为比较重点…

CF924D. Contact ATC

CF924D. Contact ATC Solution 第一波转化挺妙的。 令tlitl_itli​表示用−w-w−w的风到原点的时间&#xff0c;tritr_itri​表示用www的风到圆点的时间。 显然(i,j)(i,j)(i,j)有解的条件为(tli−tlj)(tri−trj)<0(tl_i-tl_j)(tr_i-tr_j)<0(tli​−tlj​)(tri​−trj…

Insider Dev Tour 2019 | 以技术之力,展现传承魅力

世界原本是一个漆黑的山洞的大小&#xff0c;一个“胆大妄为”的人燃起火把&#xff1b;世界原本是一个孤岛的轮廓&#xff0c;一个“野心勃勃”的人扬帆起航&#xff1b;如今&#xff0c;世界的版图已经清晰&#xff0c;是什么让我们走出黑暗与闭塞&#xff0c;从未放下对未知…

免费馅饼 逆向dp

传送门 给你n个馅饼&#xff0c;之后给出n个馅饼掉落时间和位置。一共有11个位置从 0 ~ 10&#xff0c;这人初始在第5个位置&#xff0c;让后每一秒最多能移动一个单位&#xff0c;问这个人能最多能拿多少馅饼。 思路&#xff1a; 有点像数字三角形&#xff0c;但是需要逆着推…

CF840D. Destiny

CF840D. Destiny Solution 发现问题只有询问没有修改&#xff0c;且要维护类似众数的东西&#xff0c;瞟了一眼数据范围n,m≤3e5n,m\leq 3e5n,m≤3e5&#xff0c;果断莫队&#xff0c;一点不虚&#xff0c;写到一半发现删除不好直接维护&#xff0c;用setsetset会多一个loglo…

.NET 之 ORM 性能评测

Why你应该总能听到某ORM性能比Dapper高你应该有如下疑问&#xff1a;基准测试是否权威基准测试的方式是否合理基准测试的标准是否能够统一统一基准测试标准/规范如何进行姿势正确的性能测试相信大家对 Dapper 的性能&#xff0c;以及基准测试的权威性是没有疑问的&#xff08;否…

Loj #6089. 小 Y 的背包计数问题

Loj #6089. 小 Y 的背包计数问题 Solution 似乎是比较套路的东西。 我们发现对于i≤ni\leq \sqrt ni≤n​的部分是一个多重背包&#xff0c;而剩下的部分是一个完全背包&#xff0c;因此考虑分开计算之后合并答案。 part one 当i≤ni\leq \sqrt ni≤n​时&#xff0c;令fi…

CF817E Choosing The Commander 01tire 贪心

看到集合和异或&#xff0c;可以想到01tire(但是我没有想到)。 让后就可以对于每次插入和删除一个数&#xff0c;都在01tire树上操作即可。让后记录一下到当前位(当然是从高位到低位啦)有相同前缀的数的个数。例如样例建图出来大概是这样的&#xff1a; 可以看到从编号为2的点开…

不断进化的分支和需求管理

昨天有朋友在公众号私信问我几个关于代码分支管理的问题&#xff0c;这几个问题是我去年写的《在团队中使用GitLab中的Merge Request工作模式》一文结尾时抛出的几个问题&#xff1a;如果系统上线后有紧急Bug需要处理&#xff0c;这个流程应该怎样去调整&#xff1f;每个任务都…