Предыдущая реализация содержит ошибки, например, неправильные даты для 1.0 и -1.25.Реализация ниже соответствует дате OLE, описанной в MSDN, например, здесь: https://msdn.microsoft.com/en-us/library/system.datetime.tooadate(v=vs.110).aspx
Реализация ниже соответствует документации MSDN.Он преобразует значение BigDecimal в Joda LocalDateTime.BigDecimal лучше, чем float и double, поскольку он может содержать точное значение.
class COMDateToRegularDateConverter {
private static final LocalDateTime ZERO_COM_TIME = new LocalDateTime(1899, 12, 30, 0, 0);
private static final BigDecimal MILLIS_PER_DAY = new BigDecimal(86400000);
LocalDateTime toLocalDateTime(BigDecimal comTime) {
BigDecimal daysAfterZero = comTime.setScale(0, RoundingMode.DOWN);
BigDecimal fraction = comTime.subtract(daysAfterZero).abs(); //fraction always represents the time of that day
BigDecimal fractionMillisAfterZero = fraction.multiply(MILLIS_PER_DAY).setScale(0, RoundingMode.HALF_DOWN);
return ZERO_COM_TIME.plusDays(daysAfterZero.intValue()).plusMillis(fractionMillisAfterZero.intValue());
}
}