LocalDate.ofEpochDay(X)有时会返回一个错误的或非法的结果 ,而不是抛出一个异常的X值较大。对于 实例,LocalDate.ofEpochDay(9223371671611556645L)返回日期 ,其中d.getDayOfMonth()为负值,而不是抛出 DateTimeException。
请记住,该方法now()
必须做背景的时代转换,最后调用LocalDate.ofEpochDay(...)
。所以如果你的时钟在Unix时代以毫秒为单位产生一个非常规的纪元值,那么这也会影响到now()
。而您的格式化工具只需从LocalDateTime
中取得月份中的有效日期即getDayOfMonth()
(实际上是通过TemporalAccessor
中的字段访问)。有问题的源代码:
281 public static LocalDate ofEpochDay(long epochDay) {
282 long zeroDay = epochDay + DAYS_0000_TO_1970;
283 // find the march-based year
284 zeroDay -= 60; // adjust to 0000-03-01 so leap day is at end of four year cycle
285 long adjust = 0;
286 if (zeroDay < 0) {
287 // adjust negative years to positive for calculation
288 long adjustCycles = (zeroDay + 1)/DAYS_PER_CYCLE - 1;
289 adjust = adjustCycles * 400;
290 zeroDay += -adjustCycles * DAYS_PER_CYCLE;
291 }
292 long yearEst = (400 * zeroDay + 591)/DAYS_PER_CYCLE;
293 long doyEst = zeroDay - (365 * yearEst + yearEst/4 - yearEst/100 + yearEst/400);
294 if (doyEst < 0) {
295 // fix estimate
296 yearEst--;
297 doyEst = zeroDay - (365 * yearEst + yearEst/4 - yearEst/100 + yearEst/400);
298 }
299 yearEst += adjust; // reset any negative year
300 int marchDoy0 = (int) doyEst;
301
302 // convert march-based values back to january-based
303 int marchMonth0 = (marchDoy0 * 5 + 2)/153;
304 int month = (marchMonth0 + 2) % 12 + 1;
305 int dom = marchDoy0 - (marchMonth0 * 306 + 5)/10 + 1;
306 yearEst += marchMonth0/10;
307
308 // check year now we are certain it is correct
309 int year = YEAR.checkValidIntValue(yearEst);
310 return new LocalDate(year, month, dom);
311 }
最有趣,最可疑的是,仅年被验证,而不是一个月或个月某一天的。的确,看看包含四个部分这一离奇的结果由负字符(???)分隔:
LocalDate d = LocalDate.ofEpochDay(9223371671611556645L);
System.out.println(d); // -999999999-02-0-30
System.out.println(d.getDayOfMonth()); // -30
显然,库代码被打破了一些异国情调可能由生产时代天数你的时钟不幸。 我也在Java-8中测试了相同的代码,结果相同。
更新:
到目前为止所示的LocalDate.ofEpochDay(long)
原始代码肯定是坏还因为一个事实,即存在于数字/算术溢出不检查。例如:像Long.MAX_VALUE
这样的输入会导致表达式epochDay + DAYS_0000_TO_1970
溢出并将符号更改为负值。类似地,当使用表达式400 * zeroDay
时,输入Long.MIN_VALUE
将最终溢出。我担心这不是显示代码的唯一问题。作为比较:格雷戈里尔算法的正确实施将看起来像my own time library。
旁注:
通过我的图书馆Time4J帮助我分析给定测试输入上方将产生每年远出界在threeten-BP定义,太(范围为-999999999至+ 999999999):
PlainDate date = PlainDate.of(9223371671611556645L, EpochDays.UNIX);
// java.lang.IllegalArgumentException: Year out of range: 25252733927766555
我不太清楚你能做些什么来解决问题。
第一件事是记录您的时钟产生的所有输入,将它们与观察到的3ten-bp的小车行为相关联,并且做一些研究为什么您的时钟有时会发疯。
关于threeten-bp(和Java-8!)中的错误,您可以希望threeten-bp-project团队很快就会修复它(或者说Oracle!)。导致问题的输入可能是错误的,因此您应该最好捕获异常并将时钟错误(作为根本原因)的额外消息记录下来。
什么是108795?你的约会? – sasikumar
@sasikumar:实际上它是1872095944.已更新问题。我不确定那是否是日期。你的意思是毫秒吧? – Ashwin