深入分析Date构造函数问题及解决方法

问题描述

近期在封装时间选择组件的单元测试时,为了构造出Date对象,直接使用了默认Date构造函数,
自己本地开发,测试均无问题,push远程后,某个小伙伴在本地跑测试用例时,却无法通过,具体报错截图如下
界面
通过截图信息,可以初步判断由于Date构造函数返回了不同日期导致,抱着好奇的态度查阅个各种资料后,竟然发一个小小的一个日期构造函数里面大有文章,平时自己写起来都是浅尝辄止,没有深入了解过,下面详细的介绍这个破案过程,以免各位看客后续重蹈覆辙。

问题排查

问题重现

按照一贯做法,出问题后先自己本地跑了一次测试用例,没有任何问题,初步就可以定位是开发环境问题,于是乎就看了下小伙伴nodejs版本号,版本号为6.10.0,而自己本地node版本号为10.3.0,于是直接在不同nodejs命令行下直接执行如下测试用例

const defaultDate=new Date('1995-12-17T03:24:00')
console.log(defaultDate.toString())

执行结果
Node 6.10.0
执行结果
Node 6.10.0

> const defaultDate=new Date('1995-12-17T03:24:00')
> console.log(defaultDate.toString())
Sun Dec 17 1995 11:24:00 GMT+0800 (中国标准时间)

Node 10.3.0

const defaultDate=new Date('1995-12-17T03:24:00')
undefined
console.log(defaultDate.toString())
Sun Dec 17 1995 03:24:00 GMT+0800 (中国标准时间)

到此基本确认了改问题是有Nodejs环境问题,但是为什么会有这样的问题呢,跟着我继续深入探秘下Date构造函数

深入分析

结合问题,提炼出以下小示例,以供深入分析Date构造函数

var d1 = new Date("1995/12/17 00:00:00");
var d2 = new Date("1995-12-17T00:00:00");
var d3 = new Date("1995-12-17T00:00:00Z");
console.log(d1.toString());
console.log(d2.toString());
console.log(d3.toString());

nodejs 10.3.0执行结果

> console.log(d1.toString());
Sun Dec 17 1995 00:00:00 GMT+0800 (中国标准时间)
> console.log(d2.toString());
Sun Dec 17 1995 00:00:00 GMT+0800 (中国标准时间)
> console.log(d3.toString());
Sun Dec 17 1995 08:00:00 GMT+0800 (中国标准时间)

nodejs 6.10.0执行结果

> console.log(d1.toString());
Sun Dec 17 1995 00:00:00 GMT+0800 (中国标准时间)
> console.log(d2.toString());
Sun Dec 17 1995 08:00:00 GMT+0800 (中国标准时间)
> console.log(d3.toString());
Sun Dec 17 1995 08:00:00 GMT+0800 (中国标准时间)

为什么在不同环境下Nodejs的解析行为不一样的,这就提下JS中涉及到时间的相关规范了

相关规范

ISO8601标准[参考5]

该标准指定了如果为指定偏移时间就默认为当前时间,
界面

[ES5 规范][参考6]

指出了如果没有指定偏移量,默认偏移量为Z
界面

[ES6 规范][参考7]

为了和ISO8601标准一致,又对该规范做了更改,如果时区偏移量不存在,日期时间将被解释为本地时间。
界面

源码分析

为了确认改问题由于不同规范导致,我们就需要看下V8源码里面看下的实现了。
获取不同node版本对应的v8版本号,如下图所示

//node 10.3.0
> process.versions.v8
'6.6.346.32-node.9'

//node 6.10.0
> process.versions.v8
'5.1.281.93'

查看 v8 的不同版本下git提交记录可看到在6.6版本上已经增加了对ES6规范的支持 ,实现了如果时区偏移量不存在,日期时间将被解释为本地时间。
界面

问题总结

回头看文章开头的用的日期构造函数导致的bug,就可以解释为什么”1995-12-17T00:00:00″ 在低版本下输出1995-12-17T08:00:00,而高版本下输出1995-12-17T00:00:00?

通过上述规范和源码,低版本由于会加默认偏移量Z,默认就解析成0时区的时间,而我们在东八区,所以最终我们本地的时间是1995-12-17T08:00:00,高版本下由于没有Z,默认会解析成本地时间,输出结果最终就是1995-12-17T00:00:00。

问题解决方案就是只需要加上时间偏移量即可,如下new Date(‘1995-12-17T03:24:00+08:00’)。

经验教训

由于浏览器的差异和不一致,强烈建议不要 使用Date构造函数解析日期字符串(并且Date.parse它们是等价的)。

尽可能使用“YYYY / MM / DD”作为日期字符串,或者使用年月时分秒的构造函数来构造Date对象,他们得到普遍支持和明确,有了这种格式,所有的时间都是本地的。

除非您知道自己在做什么,否则请避免使用带有连字符号的日期(“YYYY-MM-DD”),只有较新的浏览器支持它们。

参考

[1] https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Date/parse

[2]https://codereview.chromium.org/1229903004

[3]https://stackoverflow.com/questions/2587345/why-does-date-parse-give-incorrect-results/20463521#20463521

[4]https://stackoverflow.com/questions/19278797/inconsistant-date-parsing-with-missing-timezone/19279013#19279013

[5]https://en.wikipedia.org/wiki/ISO_8601

[6]http://www.ecma-international.org/ecma-262/5.1/#sec-15.9.1.15
[7]http://www.ecma-international.org/ecma-262/6.0/#sec-date-time-string-format

有兴趣欢迎关注我们的公众号:全栈探索。欢迎交流。

文章来源:

Author:开元
link:https://jdc.jd.com/archives/212575