在 Mozilla Firefox 中使用 moment.js 时获取日期中的无效日期

Getting Invalid Day in Date when using moment.js in Mozilla Firefox

我目前正在使用 moment.js 库向我的用户实时呈现日期,但经过今天的一些测试后,我注意到一些非常奇怪的事情,当在 Firefox 中向用户列出所有可能的日期时,moment.js 无法在日期“2015-09-14”中添加天数。 在搜索开发人员文档和互联网后,我发现您必须传递一个 ISO8601 有效日期或通过传递明确的日期信息来构造,但结果仍然让我感到困惑。

这里有一些代码可以理解:

工作案例(从当前时间创建一个 moment() 实例):

var printVar = moment();
var divInfo  = document.getElementById('calendars');
divInfo.innerHTML += "<ul>";

for(var i=0; i<40; i++){
    divInfo.innerHTML += "<li> " + printVar.format('DD/MM/YYYY') + " </li>";            
    printVar = moment(printVar).add(1, 'days');
}

divInfo.innerHTML += "</ul>";

我的结果:

奇怪的情况 #1(使用编码 'YYYY-MM-DD' 的时间创建 moment()):

var printVar = moment('2015-09-14');

var divInfo  = document.getElementById('calendars');
divInfo.innerHTML += "<ul>";

for(var i=0; i<40; i++){
    divInfo.innerHTML += "<li> " + printVar.format('DD/MM/YYYY') + " </li>";            
    printVar = moment(printVar).add(1, 'days');
}

divInfo.innerHTML += "</ul>";

我的结果:

奇怪的情况 #2(使用像 moment([2015, 8, 14]) 这样的显式构造函数创建 moment())return 相同的结果:

Firefox 版本:40.0.3

感谢大家的帮助!

而不是

printVar = moment(printVar.format()).add(1, 'days');

就这样

printVar.add(1, 'days');

示例代码:

var printVar, divInfo, i;

printVar = moment();
divInfo  = document.getElementById('example-1');

divInfo.innerHTML += "<ul>";
for(i=0; i<10; i++){
    divInfo.innerHTML += "<li> " + printVar.format('DD/MM/YYYY') + " </li>";            
    printVar.add(1, 'days');
}
divInfo.innerHTML += "</ul>";


printVar = moment('2015-9-10');
divInfo  = document.getElementById('example-2');
divInfo.innerHTML += "<ul>";
for(i=0; i<10; i++){
    divInfo.innerHTML += "<li> " + printVar.format('DD/MM/YYYY') + " </li>";            
    printVar.add(1, 'days');
}

divInfo.innerHTML += "</ul>";


printVar = moment([2015, 8, 14]);
divInfo  = document.getElementById('example-3');
divInfo.innerHTML += "<ul>";

for(i=0; i<10; i++){
    divInfo.innerHTML += "<li> " + printVar.format('DD/MM/YYYY') + " </li>";            
    printVar.add(1, 'days');
}

divInfo.innerHTML += "</ul>";
div {
  margin-top: 20px;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.10.6/moment.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.10.6/locale/af.js"></script>
<div id="example-1"></div>
<div id="example-2"></div>
<div id="example-3"></div>

你一定在巴西!

  • 巴西 "springs-forward" 2015 年 10 月 18 日午夜 daylight saving time。也就是说,时钟从 2015-10-17T23:59:59.999 滴答到 2015-10-18T01:00:00.000您所在时区的这一天没有午夜!

  • Moment 仍然在内部使用 Date 对象进行某些操作。

  • Date 对象在 DST 过渡边缘情况下在不同浏览器上的实现方式不同,例如处理无效或不明确的值。

  • 当您提供无效值时,
  • Firefox 会向后 偏移一个小时。其他浏览器如 Chrome shift forwards (恕我直言,这更明智)。

考虑一下我为发生夏令时的巴西(巴西利亚、圣保罗等)设置的时区:

在 FireFox 中:

但在 Chrome:

请注意,通过向后移动而不是向前移动,中间两个值的日期相差一个。

需要说明的是,这是一个浏览器问题 - 不是一时的问题。仅使用 Date 对象可以获得类似的结果。 Moment 可以 检测到此问题并进行补偿,但目前没有。

获得一致行为(使用 Datemoment)的解决方法是在处理仅日期值时使用 noon。 DST 转换始终发生在晚上,因此使用中午比午夜更安全。

火狐:

Chrome: