云顶集团官网手机版-云顶集团网站

热门关键词: 云顶集团官网手机版,云顶集团网站
竟然发现一个小小的日期构造函数里面大有文章
分类:web前端

分化Node版本引致的Date构造函数难题及减轻措施

2018/07/06 · JavaScript · Date

初稿出处: 康建云   

近年在卷入时直接受组件的单元测量检验时,为了协会出Date对象,间接利用了暗中同意Date构造函数。本身本地开辟,测量试验均无难点,push远程后,有些小同伴在本土跑测量检验用例时,却一点办法也想不出来透过,具体报错如下:

云顶集团网站 1

通过截图新闻,能够早先判定由于Date构造函数重临了分裂日期导致,抱着惊恐的态度查阅个种种资料后,竟然开采叁个纤维的日子构造函数里面不乏,平日和好写起来都以半途而返,未有深刻理解过。下边将详细介绍这一个破案进程,避防各位看客后续重蹈前辙。

JavaScript Date对象介绍

  1. 介绍

 

  Date对象,是操作日期和时间的对象。Date对象对日期和岁月的操作只可以通过艺术。

 

  1. 构造函数

 

2.1 new Date() :重临当前的本地日期和时间

参数:无

 

返回值:

 

{Date} 重返三个意味本地日期和时间的Date对象。

 

示例:

 

 

var dt = new Date();

console.log(dt); // => 重临贰个意味着本地日期和时间的Date对象

 

 

2.2 new Date(milliseconds) :把纳秒数转变为Date对象

参数:

 

①milliseconds {int} :皮秒数;表示从'一九六七/01/01 00:00:00'为起源,开头增大的阿秒数。

 

只顾:起源的时分秒还要加上圈套前所在的时区,香港(Hong Kong卡塔 尔(阿拉伯语:قطر‎时间的时区为东8区,源点时间实在为:'一九六八/01/01 08:00:00'

 

返回值:

 

{Date} 再次回到多个叠合后的Date对象。

 

示例:

 

 

var dt = new Date(1000 * 60 * 1); // 前行1分钟的微秒数

console.log(dt); // => {Date}:1970/01/01 08:01:00

dt = new Date(-1000 * 60 * 1); // 倒退1分钟的阿秒数

console.log(dt); // => {Date}:1970/01/01 07:59:00

 

 

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

参数:

 

①dateStr {string} :可更改为Date对象的字符串(可归纳时间);字符串的格式重要有三种:

 

1) yyyy/MM/dd HH:mm:ss (推荐卡塔 尔(英语:State of Qatar):若省略时间,重返的Date对象的日子为 00:00:00。

 

2) yyyy-MM-dd HH:mm:ss :若省略时间,重临的Date对象的岁月为 08:00:00(加上本地时区)。若不省略时间,此字符串在IE中会调换战败!

 

返回值:

 

{Date} 重回两个转换后的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}:2015-12-25 08:00:00 (加上了东8区的时区)

dt = new Date('二零一四-12-25 12:00:00'); // yyyy-MM-dd HH:mm:ss (注意:此转变情势在IE中会报错!)

console.log(dt); // => {Date}:2014-12-25 12:00:00

 

 

2.4 new Date(year, month, opt_day, opt_hours, opt_minutes, opt_seconds, opt_milliseconds) :把年月日、时分秒调换为Date对象

参数:

 

①year {int} :年份;4位数字。如:1999、2014

 

②month {int} :月份;2位数字。从0初始思忖,0象征六月份、11代表七月份。

 

③opt_day {int} 可选:号; 2位数字;从1发轫总结,1象征1号。

 

④opt_hours {int} 可选:时;2位数字;取值0~23。

 

⑤opt_minutes {int} 可选:分;2位数字;取值0~59。

 

⑥opt_seconds {int} 可选:秒;2未数字;取值0~59。

 

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

 

返回值:

 

{Date} 重临叁个调换后的Date对象。

 

示例:

 

var dt = new Date(贰零壹肆, 11); // 二〇一六年1一月(这里输入的月份数字为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(二〇一五, 12, 25); // 2016年17月15日(这里输入的月份数字为12,表示第14个月,跳转到第二年的10月)

console.log(dt); // => {Date}:2015/01/25

 

 

  1. 属性

 

无;Date对象对日期和时间的操作只好通过艺术。

 

 

 

  1. 实例方法

 

  Date对象的实例方法主要分为2种植花朵样:本地时间和UTC时间。同贰个情势,日常都会有此2种时光格式操作(方法名带UTC的,正是操作UTC时间),这里最首要介绍对本土时间的操作。

 

 

 

4.1 get方法

4.1.1 getFullYear() :再次来到Date对象的年份值;4位年份。

 

4.1.2 getMonth() :重回Date对象的月度值。从0初叶,所以真实月份=重临值+1 。

 

4.1.3 getDate() :再次回到Date对象的月份中的日期值;值的范围1~31 。

 

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

 

4.1.5 getMinutes() :重返Date对象的分钟值。

 

4.1.6 getSeconds() :再次来到Date对象的秒数值。

 

4.1.7 getMilliseconds() :再次来到Date对象的纳秒值。

 

4.1.8 getDay() :重临Date对象的三日中的星期值;0为周日,1为礼拜一、2为星期三,依此类推

 

4.1.9 getTime() :重临Date对象与'一九七〇/01/01 00:00:00'之间的纳秒值(Hong Kong时间的时区为东8区,起源时间实际上为:'一九七零/01/01 08:00:00') 。

 

示例:

 

dt.getFullYear(); // => 2014:年

dt.getMonth(); // => 11:月;实际为四月份(月份从0最初总计)

dt.getDate(); // => 25:日

dt.getHours(); // => 15:时

dt.getMinutes(); // => 30:分

dt.getSeconds(); // => 40:秒

dt.getMilliseconds(); // => 333:毫秒

dt.getDay(); // => 4:星期几的值

dt.getTime(); // => 14一九五零2640333 :再次回到Date对象与'1966/01/01 00:00:00'之间的纳秒值(东方之珠时间的时区为东8区,起源时间实在为:'一九六七/01/01 08:00:00')

 

 

4.2 set方法

4.2.1 setFullYear(year, opt_month, opt_date) :设置Date对象的年份值;4位年份。

 

4.2.2 setMonth(month, opt_date) :设置Date对象的月份值。0代表五月,11代表四月。

 

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

 

4.2.4 setHours(hour, opt_min, opt_sec, opt_msec) :设置Date对象的时辰值。

 

4.2.5 setMinutes(min, opt_sec, opt_msec) :设置Date对象的分钟值。

 

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

 

4.2.7 setMilliseconds(msec) :设置Date对象的阿秒值。

 

示例:

 

var dt = new Date();

dt.setFullYear(2014); // => 2014:年

dt.setMonth(11); // => 11:月;实际为七月份(月份从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毫秒

 

 

4.3 其余艺术

4.3.1 toString() :将Date调换为三个'年月日 时分秒'字符串

 

4.3.2 toLocaleString() :将Date转变为一个'年月日 时分秒'的本土格式字符串

 

4.3.3 toDateString() :将Date转换为一个'年月日'字符串

 

4.3.4 toLocaleDateString() :将Date转变为多个'年月日'的本地格式字符串

 

4.3.5 toTimeString() :将Date调换为叁个'时分秒'字符串

 

4.3.6 toLocale提姆eString() :将Date调换为三个'时分秒'之处格式字符串

 

4.3.7 valueOf() :与getTime()同样, 重返Date对象与'壹玖陆柒/01/01 00:00:00'之间的纳秒值(法国巴黎时间的时区为东8区,起源时间实在为:'一九六九/01/01 08:00:00') 

 

示例:

 

var dt = new Date();

console.log(dt.toString()); // => Tue Dec 23 二零一五 22:56:11 青霉素T+0800 (中国标按时间) :将Date转变为三个'年月日 时分秒'字符串

console.log(dt.toLocaleString()); // => 2015年5月16日 上午10:56:11  :将Date调换为四个'年月日 时分秒'的地面格式字符串

 

console.log(dt.toDateString()); // => Tue Dec 23 二零一四:将Date调换为三个'年月日'字符串

console.log(dt.toLocaleDateString()); // => 2015年四月10日:将Date转变为贰个'年月日'的本土格式字符串

 

console.log(dt.to提姆eString()); // => 22:56:11 克拉霉素T+0800 (中国标准时期) :将Date转变为四个'时分秒'字符串

console.log(dt.toLocaleTimeString()); // => 早晨10:56:11 :将Date调换为四个'时分秒'的当地格式字符串

 

console.log(dt.valueOf()); // => 重回Date对象与'一九七零/01/01 00:00:00'之间的微秒值(时尚之都时间的时区为东8区,源点时间实际上为:'壹玖陆捌/01/01 08:00:00')

 

 

  1. 静态方法

 

5.1 Date.now()

申明:重返当前些天子和岁月的Date对象与'一九六六/01/01 00:00:00'之间的纳秒值(新加坡时间的时区为东8区,起源时间实在为:'一九六九/01/01 08:00:00') 

 

参数:无

 

返回值:

 

{int} :当前时刻与开始时间之内的阿秒数。

 

示例:

 

 

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

  

 

5.2 Date.parse(dateStr)

评释:把字符串调换为Date对象 ,然后回来此Date对象与'一九六九/01/01 00:00:00'之间的皮秒值(新加坡时间的时区为东8区,起源时间莫过于为:'一九七零/01/01 08:00:00')

 

参数:

 

①dateStr {string} :可转移为Date对象的字符串(可粗略时间);字符串的格式首要有二种:

 

1) yyyy/MM/dd HH:mm:ss (推荐卡塔 尔(英语:State of Qatar):若省略时间,重临的Date对象的流年为 00:00:00。

 

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

 

返回值:

 

{int} 再次来到调换后的Date对象与早先时间之间的微秒数。

 

示例:

 

 

console.log(Date.parse('2014/12/25 12:00:00')); // => 1419480000000

console.log(Date.parse('贰零壹肆-12-25 12:00:00')); // => 14一九四九0000000  (注意:此转变格局在IE中回到NaN!)

 

 

  1. 实际操作

 

6.1 C#的DateTime类型调换为Js的Date对象

说明:C#的DateTime类型通过Json连串化再次来到给前台的格式为:"/Date(1419492640000)/" 。中间的数字,表示DateTime的值与早先时间之内的微秒数。

 

示例:

 

后台代码:简单的ashx

 

 

public void ProcessRequest (HttpContext context) {

    System.Web.Script.Serialization.JavaScriptSerializer js = new System.Web.Script.Serialization.JavaScriptSerializer();

    DateTime dt = DateTime.Parse("2014-12-25 15:30:40");

    string rs = js.Serialize(dt); // 系列化成Json

    context.Response.ContentType = "text/plain";

    context.Response.Write(rs);

}

前台代码:

 

 

var dateTimeJsonStr = '/Date(1419492640000)/'; // C# DateTime类型转变的Json格式

var msecStr = dateTimeJsonStr.toString().replace(//Date(([-]?d+))//gi, "$1"); // => '1419502640000' :通过正则替换,获取阿秒字符串

var msesInt = Number.parseInt(msecStr); // 飞秒字符串调换来数值

var dt = new Date(msesInt); // 初始化Date对象

console.log(dt.toLocaleString()); // => 2014年12月25日 下午3:30:40

 

 

6.2  获取倒计时

注脚:计算当前时刻离指标时间相差多少天时分。

 

示例:

 

/**

* 重回倒计时

* @param dt {Date}:目的Date对象

* @return {Strin} :重回倒计时:X天X时X分

*/

function getDownTime(dt) {

    // 1.获得倒计时

    var intervalMsec = dt - Date.now(); // 指标时间减去以往的时间,获取两岸相距的飞秒数

    var intervalSec = intervalMsec / 1000; // 调换来秒数

    var day = parseInt(intervalSec / 3600 / 24); // 天数

    var hour = parseInt((intervalSec - day * 24 * 3600) / 3600); // 小时

    var min = parseInt((intervalSec - day * 24 * 3600 - hour * 3600) / 60); // 分钟

 

    // 2.若相距的纳秒小于0 ,表示目标时间低于当今天子,那时的取的值都以负的:-X天-时-分,展现时,只体现天数前边为负的就能够。

    if (intervalMsec < 0) {

        hour = 0 - hour;

        min = 0 - min;

    }

 

    // 3.拼接字符串并赶回

    var rs = day + '天' + hour + '时' + min + '分';

    return rs;

}

 

竟然发现一个小小的日期构造函数里面大有文章。// 当前光阴:二零一五/12/28 13:26

console.log(getDownTime(new Date('2015/06/01'))); // => 154天10时33分

console.log(getDownTime(new Date('2014/01/01'))); // => -361天13时26分

 

 

6.3 比较2个Date对象的朗朗上口

声明:能够对照2者的与初始时间的阿秒数,来差别轻重缓急。

 

示例:

 

var dt1 = new Date('2015/12/01');

var dt2 = new Date('2015/12/25');

console.log(dt1 > dt2); // => false

 

Date对象介绍 1. 介绍 Date对象,是操作日期和时间的对象。Date对象对日期和岁月的操作只可以通过措施。

  1. 构造函数 2.1 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('一九九五-12-17T03:24:00') > console.log(defaultDate.toString()) Sun Dec 17 一九九五 11:24:00 威斯他霉素T +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('一九九五-12-17T03:24:00') undefined console.log(defaultDatae.toString()) Sun Dec 17 1991 03:24:00 博来霉素T+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构造函数:

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 1991 00:00:00 威他霉素T+0800 (中黄炎子孙民共和国家规范定时间) > console.log(d2.toString()); Sun Dec 17 1994 00:00:00 丙胺搏来霉素T+0800 (中夏族民共和国家规范准时期) > console.log(d3.toString()); Sun Dec 17 199508:00:00 维生霉素T+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 一九九五 00:00:00 螺旋霉素T+0800 (中华夏儿女民共和国家规范定期间) > console.log(d2.toString()); Sun Dec 17 1995 08:00:00 罗红霉素T+0800 (中中原人民共和国家标准定时期) > console.log(d3.toString()); Sun Dec 17 199208:00:00 地霉素T+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中提到届期刻的相干标准了。

连锁标准

ISO8601标准[参考5]

该专门的学业钦定了一旦为钦赐偏移时间就默以为当前时刻。

云顶集团网站 2

[ES5 规范][参考6]

提议了一旦未有一些名偏移量,暗许偏移量为Z。

云顶集团网站 3

[ES6 规范][参考7]

为了和ISO8601标准意气风发致,又对该职业做了校正,假诺时区偏移量不设有,日期时间将被解释为本地时间。

云顶集团网站 4

源码解析

为了确认该难题是出于分化职业招致的,咱们就要求看下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正规的补助,完成了后生可畏旦时区偏移量不设有,日期时间将被分解为本地时间的效率。

云顶集团网站 5

难题总括

回头看文章开端的用的日期构造函数招致的bug,就足以分解”壹玖玖伍-12-17T00:00:00″ 在低版本下输出一九九五-12-17T08:00:00,而高版本下输出1992-12-17T00:00:00的标题了。

通过上述标准和源码,低版本由于会加私下认可偏移量Z,私下认可就分析成0时区的小运,而大家在东八区,所以最终大家本地的岁月是1994-12-17T08:00:00,高版本下由于并未有Z,私下认可会深入分析花费地时间,输出结果最终便是1991-12-17T00:00:00。

难点解决方案正是只必要丰硕岁月偏移量就能够,如下new Date(‘一九九一-12-17T03:24:00+08:00’)。

经验教诲

出于浏览器的异样和不平等,猛烈提出不要 使用Date构造函数剖判日期字符串(並且Date.parse它们是等价的卡塔尔国。

尽恐怕使用“YYYY / MM / DD”作为日期字符串,或许采取时间时分秒的构造函数来布局Date对象,他们获取普及地援助。有了这种格式,全体的时间都以本地的。

唯有你明白本身在做如何,不然请幸免接受含有连字符号的日子(”YYYY-MM-DD”卡塔 尔(阿拉伯语:قطر‎,独有较新的浏览器扶植它们。

参考

[1]

[2]

[3]

[4]

[5]

[6]

[7]

1 赞 1 收藏 评论

云顶集团网站 6

本文由云顶集团官网手机版发布于web前端,转载请注明出处:竟然发现一个小小的日期构造函数里面大有文章

上一篇:原文出处,那什么是自由变量呢 下一篇:没有了
猜你喜欢
热门排行
精彩图文