20

尝试使用 GregorianCalendar 时,我在计算自特定日期以来的天数时陷入了奇点。在 scala 解释器中,我输入:

scala>import java.util.GregorianCalendar
scala>import java.util.Calendar
scala>val dateToday = new GregorianCalendar(2012,Calendar.MAY,22).getTimeInMillis()
dateToday: Long = 1337637600000
scala>val days1 = (dateToday - (new GregorianCalendar(1976,Calendar.MARCH,28).getTimeInMillis())) / (1000*3600*24)
days1: Long = 13203
scala>val days2 = (dateToday - (new GregorianCalendar(1976,Calendar.MARCH,29).getTimeInMillis())) / (1000*3600*24)
days2: Long = 13203

我不知道 1976 年是闰年这一事实是否重要,但 days1 和 days2 应该被 1 隔开。这是自 1970 年以来历史上唯一发生这种奇点的时刻。

想知道发生了什么,我计算了前面提到的两个日期之间的差异,它只给了我 23 小时的差异!那天发生了什么?维基百科显然对此只字未提。

更重要的是,如何计算自特定日期以来的实际天数?

4

3 回答 3

46

问题

夏令时只有 23 小时。

根据这个1976 年 3 月 28 日,至少在巴黎是夏令时。那天凌晨 1 点到 2 点之间的时间根本不存在。

虽然它必须是一个区域设置问题,因为在我的计算机上我做对了:

scala> val days1 = (dateToday - (new GregorianCalendar(1976, Calendar.MARCH, 28).getTimeInMillis())) / (1000 * 3600 * 24)
days1: Long = 13203

scala> val days2 = (dateToday - (new GregorianCalendar(1976, Calendar.MARCH, 29).getTimeInMillis())) / (1000 * 3600 * 24)
days2: Long = 13202

那天我不在巴黎或其他任何时间变化的地方;时间在我所在的 1976 年 4 月 25 日改变了。所以我在那个日期得到了“跳过的一天”行为:

scala> val days3 = (dateToday - (new GregorianCalendar(1976, Calendar.APRIL, 25).getTimeInMillis())) / (1000 * 3600 * 24)
days3: Long = 13175

scala> val days4 = (dateToday - (new GregorianCalendar(1976, Calendar.APRIL, 26).getTimeInMillis())) / (1000 * 3600 * 24)
days4: Long = 13175

Erwin 在评论中指出,您在这里唯一注意到不正确日期差异的可能原因是,所有其他夏令时都被那些年份中也发生的 25 小时制所抵消,当时夏令时已得到纠正。

解决方案

使用更好的库进行日期处理。joda -time库正确地做到了(以及整体上更好的日期/时间框架):

import org.joda.time.Days
import org.joda.time.DateTimeConstants
import org.joda.time.DateMidnight

val d1 = new DateMidnight(1976, DateTimeConstants.APRIL, 25)
val d2 = new DateMidnight(1976, DateTimeConstants.APRIL, 26)
val x = Days.daysBetween(d1, d2).getDays()
println(x) // 1
于 2012-05-22T08:38:33.600 回答
4

如果浮动除法选项不可用,我能想到的最佳答案是在计算两天之间的差异时增加一小时(1000 * 3600):

scala> (dateToday - (new GregorianCalendar(1976, Calendar.MARCH, 28).getTimeInMillis()) + 1000*3600) / (1000 * 3600 * 24)
days1: Long = 13204
scala> (dateToday - (new GregorianCalendar(1976, Calendar.MARCH, 29).getTimeInMillis()) + 1000*3600) / (1000 * 3600 * 24)
days1: Long = 13203

它应该适用于每个日期,因为您不再受闰小时的影响。

于 2012-05-22T09:18:37.903 回答
3

dhg 指出了我的想法:这一天是“夏令时”的一天。由于这一天只有 23 小时,欧几里得除以一天等于 0。

实际上,使用 GregorianCalendar 对象只是将日期用作毫秒,因此除以一天作为整数只会截断结果。

尝试进行浮点除法,而不是欧几里得(整数)除法,然后将结果四舍五入。

于 2012-05-22T08:54:10.153 回答