不同Node版本导致的Date构造函数问题及解决方法

深入分析

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

JavaScript

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());

1
2
3
4
5
6
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执行结果:

JavaScript

> 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 (中国标准时间)

1
2
3
4
5
6
> 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执行结果:

JavaScript

> 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 (中国标准时间)

1
2
3
4
5
6
> 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中涉及到时间的相关规范了。

Date.UTC() 是一种静态方法。Date.UTC() 方法的参数指定日期和时间,它们都是
UTC 时间,处于 GMT 时区。指定的 UTC 时间将转换成毫秒的形式,这样构造函数
Date() 和方法 Date.setTime() 就可以使用它了。

4.3.4 toLocaleDateString() :将Date转换为一个’年月日’的本地格式字符串

相关规范

ISO8601标准[参考5]

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

图片 1

[ES5 规范][参考6]

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

图片 2

[ES6 规范][参考7]

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

图片 3

创建 Date 对象的语法:

5.2 Date.parse(dateStr)
说明:把字符串转换为Date对象 ,然后返回此Date对象与’1970/01/01
00:00:00’之间的毫秒值(北京时间的时区为东8区,起点时间实际为:’1970/01/01
08:00:00′)

问题总结

回头看文章开头的用的日期构造函数导致的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’)。

var d = new Date();

示例:

问题排查

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

JavaScript

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

1
2
3
const defaultDate = new Date(‘1995-12-17T03:24:00’);
 
console.log(defaultDate.toString());

执行结果,

Node 6.10.0:

JavaScript

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

1
2
3
4
> 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:

JavaScript

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

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

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

Date.UTC(year, month, day, hours, minutes, seconds, ms)

复制代码 代码如下:

源码分析

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

JavaScript

//node 10.3.0 > process.versions.v8 ‘6.6.346.32-node.9’ //node 6.10.0
> process.versions.v8 ‘5.1.281.93’

1
2
3
4
5
6
7
//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规范的支持
,实现了如果时区偏移量不存在,日期时间将被解释为本地时间的效果。

图片 4

复制代码 代码如下:

⑦opt_milliseconds {int} 可选:毫秒;取值0~999。

参考

[1]

[2]

[3]

[4]

[5]

[6]

[7]

1 赞 1 收藏
评论

图片 5

复制代码 代码如下:

注意:起点的时分秒还要加上当前所在的时区,北京时间的时区为东8区,起点时间实际为:’1970/01/01
08:00:00′

经验教训

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

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

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

setDate() 设置 Date 对象中月的某一天 (1 ~ 31)。
setMonth() 设置 Date 对象中月份 (0 ~ 11)。
setFullYear() 设置 Date
对象中的年份(四位数字)。注意不要使用setYear()方法。
setHours() 设置 Date 对象中的小时 (0 ~ 23)。
setMinutes() 设置 Date 对象中的分钟 (0 ~ 59)。
setSeconds() 设置 Date 对象中的秒钟 (0 ~ 59)。
setMilliseconds() 设置 Date 对象中的毫秒 (0 ~ 999)。
setTime() 以毫秒设置 Date 对象。
setUTCDate() 根据世界时设置 Date 对象中月份的一天 (1 ~ 31)。
setUTCMonth() 根据世界时设置 Date 对象中的月份 (0 ~ 11)。
setUTCFullYear() 根据世界时设置 Date 对象中的年份(四位数字)。
setUTCHours() 根据世界时设置 Date 对象中的小时 (0 ~ 23)。
setUTCMinutes() 根据世界时设置 Date 对象中的分钟 (0 ~ 59)。
setUTCSeconds() 根据世界时设置 Date 对象中的秒钟 (0 ~ 59)。
setUTCMilliseconds() 根据世界时设置 Date 对象中的毫秒 (0 ~ 999)。

4.2.6 setSeconds(sec, opt_msec) :设置Date对象的秒数值。

不同Node版本导致的Date构造函数问题及解决方法

2018/07/06 · JavaScript
· Date

原文出处:
康建云   

近期在封装时间选择组件的单元测试时,为了构造出Date对象,直接使用了默认Date构造函数。自己本地开发,测试均无问题,push远程后,某个小伙伴在本地跑测试用例时,却无法通过,具体报错如下:

图片 6

通过截图信息,可以初步判断由于Date构造函数返回了不同日期导致,抱着好奇的态度查阅个各种资料后,竟然发现一个小小的日期构造函数里面大有文章,平时自己写起来都是浅尝辄止,没有深入了解过。下面将详细介绍这个破案过程,以免各位看客后续重蹈覆辙。

console.log(d1);  //Tue Mar 13 2012 00:00:00 GMT+0800 (中国标准时间)
console.log(d2);  //Tue Mar 13 2012 08:00:00 GMT+0800 (中国标准时间)
console.log(d3);  //Tue Mar 13 2012 00:00:00 GMT+0800 (中国标准时间)

参数:无

set系列方法

4.1.4 getHours() :返回Date对象的小时值。

为了避免这些问题,请遵照下面的建议:

示例:

Date()

4.2.2 setMonth(month, opt_date)
:设置Date对象的月份值。0表示1月,11表示12月。

ECMAScript5添加了Data.now()方法,返回表示调用这个方法时的日期和时间的毫秒数。IE9才开始支持,不过我们可以通过

③opt_day {int} 可选:号; 2位数字;从1开始计算,1表示1号。

//Date 对象会自动把当前日期和时间保存为其初始值。
new Date();

4.1.9 getTime() :返回Date对象与’1970/01/01
00:00:00’之间的毫秒值(北京时间的时区为东8区,起点时间实际为:’1970/01/01
08:00:00′) 。

复制代码 代码如下:

①dateStr {string}
:可转换为Date对象的字符串(可省略时间);字符串的格式主要有两种:

var d1 = Date.now();
var d2 = + new Date();

4.1.6 getSeconds() :返回Date对象的秒数值。

更多信息可以参考:JavaScript and Dates, What a
Mess!和
SO中的讨论

复制代码 代码如下:

//year-年:代表年份的整数值。为了避免2000年问题最好指定4位数的年份;
使用1998, 而不要用98
//month-月:代表月份的整数值从0(1月)到11(12月)
//day-日:代表一个月中的第几天的整数值,从1开始
//hour-小时:代表一天中的小时数的整数值 (24小时制)
//minute-分钟
//second-秒
//millisecond-毫秒
new Date(year, month, day [, hour, minute, second, millisecond]);

复制代码 代码如下:

console.log(d1);  //1379393793104
console.log(d2);  //1379393793104

4.2.3 setDate(date) :设置Date对象的月份中的日期值;值的范围1~31 。

//dateString-日期字符串:表示日期的字符串值。此字符串应该是在parse方法中识别的格式。
new Date(dateString);

复制代码 代码如下:

您可能感兴趣的文章:

  • 扩展JS
    Date对象时间格式化功能的小例子
  • JavaScript Date对象详解
  • 详解JavaScript对Date对象的操作问题(生成一个倒数7天的数组)
  • javascript转换日期字符串为Date日期对象的方法
  • 浅谈JavaScript中Date(日期对象),Math对象
  • 浅谈JavaScript
    Date日期和时间对象
  • JavaScript
    中Date对象的格式化代码方法汇总

{Date} 返回一个叠加后的Date对象。

toString系列方法

var dt = new Date();
console.log(dt); // => 返回一个表示本地日期和时间的Date对象

注意:parse()日期和时间都基于本地时区而非GMT来创建。而UTC()是基于GMT来创建。他们的参数也不相同。

复制代码 代码如下:

var d1 = new Date(“2012/03/13”);
var d2 = new Date(“2012-03-13”);
var d3 = new Date(“2012-3-13”);

{Date} 返回一个转换后的Date对象。

Date() 方法可返回当天的日期和时间。

2.3 new Date(dateStr) :把字符串转换为Date对象
参数:

toString() 把 Date
对象转换为字符串,toString()总是返回一个用美式英语表达的字符串。
toTimeString() 把 Date 对象的时间部分转换为字符串。
toDateString() 把 Date 对象的日期部分转换为字符串。
toUTCString() 根据世界时,把 Date 对象转换为字符串。
toLocaleString() 根据本地时间格式,把 Date 对象转换为字符串。
toLocaleTimeString() 根据本地时间格式,把 Date
对象的时间部分转换为字符串。
toLocaleDateString() 根据本地时间格式,把 Date
对象的日期部分转换为字符串。

var dt = new Date(‘2014/12/25’); // yyyy/MM/dd
console.log(dt); // => {Date}:2014/12/25 00:00:00
dt = new Date(‘2014/12/25 12:00:00’); // yyyy/MM/dd HH:mm:ss
console.log(dt); // => {Date}:2014/12/25 12:00:00
dt = new Date(‘2014-12-25’); // yyyy-MM-dd
console.log(dt); // => {Date}:2014-12-25 08:00:00
(加上了东8区的时区)
dt = new Date(‘2014-12-25 12:00:00’); // yyyy-MM-dd HH:mm:ss
(注意:此转换方式在IE中会报错!)
console.log(dt); // => {Date}:2014-12-25 12:00:00

复制代码 代码如下:

复制代码 代码如下:

复制代码 代码如下:

2) yyyy-MM-dd HH:mm:ss :若省略时间,返回的Date对象的时间为
08:00:00(加上本地时区)。若不省略时间,此字符串在IE中返回NaN(非数字)!

UTC()

示例:

getDate() 从 Date 对象返回一个月中的某一天 (1 ~ 31)。
getDay() 从 Date 对象返回一周中的某一天 (0 ~ 6)。
getMonth() 从 Date 对象返回月份 (0 ~ 11)。
getFullYear() 从 Date 对象以四位数字返回年份。注意不要使用getYear()。
getHours() 返回 Date 对象的小时 (0 ~ 23)。
getMinutes() 返回 Date 对象的分钟 (0 ~ 59)。
getSeconds() 返回 Date 对象的秒数 (0 ~ 59)。
getMilliseconds() 返回 Date 对象的毫秒(0 ~ 999)。
getTime() 返回 1970 年 1 月 1 日至今的毫秒数。
getTimezoneOffset() 返回本地时间与格林威治标准时间 (GMT) 的分钟差。
getUTCDate() 根据世界时从 Date 对象返回月中的一天 (1 ~ 31)。
getUTCDay() 根据世界时从 Date 对象返回周中的一天 (0 ~ 6)。
getUTCMonth() 根据世界时从 Date 对象返回月份 (0 ~ 11)。
getUTCFullYear() 根据世界时从 Date 对象返回四位数的年份。
getUTCHours() 根据世界时返回 Date 对象的小时 (0 ~ 23)。
getUTCMinutes() 根据世界时返回 Date 对象的分钟 (0 ~ 59)。
getUTCSeconds() 根据世界时返回 Date 对象的秒钟 (0 ~ 59)。
getUTCMilliseconds() 根据世界时返回 Date 对象的毫秒(0 ~ 999)。

var dt = new Date();
dt.setFullYear(2014); // => 2014:年
dt.setMonth(11); // => 11:月;实际为12月份(月份从0开始计算)
dt.setDate(25); // => 25:日
dt.setHours(15); // => 15:时
dt.setMinutes(30); // => 30:分
dt.setSeconds(40); // => 40:秒
dt.setMilliseconds(333); // => 333:毫秒
console.log(dt); // =>  2014年12月25日 15点30分40秒 333毫秒

ECMAScript中的Date类型是早期Java中的Java.util.Date类基础上构建的。为此,Date类型使用来自UTC(Coordinated
Universal
Time,国际协调时间)1970年1月1日午夜(零时)开始经过的毫秒数来保存日期。在使用这种数据存储格式的条件下,Date类型保存的日期能够精确到1970年1月1日之前或之后的285616年。

示例:

console.log(d.toLocaleString(“ko-KR”));  //2013년 9월 17일 오후
1:48:24

2.1 new Date() :返回当前的本地日期和时间
参数:无

JavaScript中的日期转换非常的诡异,不仅会因为参数不同而有不同的解释结果,而且在各个浏览器中的表现也有所不同,如下:

示例:

console.log(d);                      //Tue Sep 17 2013 13:37:04 GMT+0800
(中国标准时间)
console.log(d.toString());           //Tue Sep 17 2013 13:37:04 GMT+0800
(中国标准时间)
console.log(d.toTimeString());       //13:37:04 GMT+0800
(中国标准时间)
console.log(d.toDateString() );      //Tue Sep 17 2013
console.log(d.toUTCString());        //Tue, 17 Sep 2013 05:37:04 GMT
console.log(d.toLocaleString());     //2013年9月17日 下午1:37:04
console.log(d.toLocaleTimeString()); //下午1:37:04
console.log(d.toLocaleDateString()); //2013年9月17日

console.log(Date.now()); // => 1419431519276

parse()

4.1 get方法
4.1.1 getFullYear() :返回Date对象的年份值;4位年份。

//value-毫秒:代表自世界协调时1970年1月1日00:00:00开始的数值。
new Date(value);

参数:

var d = new Date();

您可能感兴趣的文章:

  • 扩展JS
    Date对象时间格式化功能的小例子
  • JavaScript Date对象详解
  • 详解JavaScript对Date对象的操作问题(生成一个倒数7天的数组)
  • JavaScript原生对象之Date对象的属性和方法详解
  • javascript转换日期字符串为Date日期对象的方法
  • 浅谈JavaScript中Date(日期对象),Math对象
  • JavaScript
    中Date对象的格式化代码方法汇总

parse() 方法可解析一个日期时间字符串,并返回 1970/1/1
午夜距离该日期时间的毫秒数。

var dt = new Date(2014, 11); // 2014年12月(这里输入的月份数字为11)
console.log(dt); // => {Date}:2014/12/01 00:00:00
dt = new Date(2014, 11, 25); // 2014年12月25日
console.log(dt); // => {Date}:2014/12/25 00:00:00
dt = new Date(2014, 11, 25, 15, 30, 40); // 2014年12月25日
15点30分40秒
console.log(dt); // => {Date}:2014/12/25 15:30:40
dt = new Date(2014, 12, 25); //
2014年13月25日(这里输入的月份数字为12,表示第13个月,跳转到第二年的1月)
console.log(dt); // => {Date}:2015/01/25

1.坚持使用 ”YYYY/MM/DD” 的日期字符串格式
2.避免使用带有连字号的日期字符串格式”YYYY-MM-DD”
3.要指定四位数的年份
4.Chrome浏览器的比其他浏览器能接受更多的日期字符串,所以如果在Chrome浏览器没有问题,不代表其他浏览器没有问题

②month {int} :月份;2位数字。从0开始计算,0表示1月份、11表示12月份。

相关文章

发表评论

电子邮件地址不会被公开。 必填项已用*标注

*
*
Website