我遇到了一种情况,我想将儒略日期转换为 java.time.Instant (如果有意义),或者将Java时间转换为更容易理解的时间。我对儒略日期的理解来自阅读Wikipedia page。有一堆不同的variants,我尝试读取的日期使用的历元不同于所有这些。

例如,假设纪元是Calendar (New Style) Act 1750的开始,儒略历日期是95906.27600694445,在这种情况下,我认为是 CE 2015 April 15 06:37:26.9 UT ,如何从中得到一个瞬间?稍后需要调整时区。

我注意到有一个名为 JulianFields 的类,但是我不知道在哪里/如何使用它。另外,我在程序包中看到的大多数方法都使用intlong,而对于double并没有任何使用。

因此,有没有一种简单的方法可以将使用不同历元的儒略日期转换为Java 8 Instant(如果我的想法是错误的,则可以转换为其他时间)。

最佳答案

这是使用新的Java 8类的解决方案:

public class JulianDay {
    private static final double NANOS_PER_DAY = 24.0 * 60.0 * 60.0 * 1000000000.0;

    // Calculate Instants for some epochs as defined in Wikipedia.
    public static final Instant REDUCED_JD =
            ZonedDateTime.of(1858, 11, 16, 12, 0, 0, 0, ZoneOffset.UTC).toInstant();
    public static final Instant MODIFIED_JD =
            ZonedDateTime.of(1858, 11, 17, 0, 0, 0, 0, ZoneOffset.UTC).toInstant();
    public static final Instant JULIAN_DATE =
            REDUCED_JD.minus(2400000, ChronoUnit.DAYS);

    private final Instant epoch;

    public JulianDay(Instant epoch) {
        super();
        this.epoch = epoch;
    }

    public Instant toInstant(double day) {
        long l = (long) day;
        return epoch
                .plus(l, ChronoUnit.DAYS)
                .plusNanos(Math.round((day - l) * NANOS_PER_DAY));
    }

    public static void main(String[] args) {
        // Use the example values from Wikipedia for 2015-09-07 13:21 UTC.
        System.out.println(new JulianDay(REDUCED_JD).toInstant(57273.05625));
        // Output: 2015-09-07T13:21:00.000000126Z
        System.out.println(new JulianDay(MODIFIED_JD).toInstant(57272.55625));
        // Output: 2015-09-07T13:21:00.000000126Z
        System.out.println(new JulianDay(JULIAN_DATE).toInstant(2457273.05625));
        // Output: 2015-09-07T13:20:59.999991953Z
    }
}

关于您询问的JulianFields,您可以定义一个自定义格式化程序,如下所示:
DateTimeFormatter formatter = new DateTimeFormatterBuilder()
    .appendValue(JulianFields.MODIFIED_JULIAN_DAY)
    .toFormatter().withZone(ZoneOffset.UTC);

不幸的是,它不支持几天的时间:
System.out.println(formatter.format(Instant.now())); // Output: 57249
System.out.println(LocalDate.from(formatter.parse("57249"))); // Output: 2015-08-15

08-08 01:02