1. ホーム
  2. java

[解決済み] Java 8: 2つのLocalDateTimeの複数単位での差分

2022-03-16 07:24:13

質問

2つの LocalDateTime .

出力は以下の形式である必要があります。 y years m months d days h hours m minutes s seconds . 以下は、私が書いたものです。

import java.time.Duration;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.Period;
import java.time.ZoneId;

public class Main {

    static final int MINUTES_PER_HOUR = 60;
    static final int SECONDS_PER_MINUTE = 60;
    static final int SECONDS_PER_HOUR = SECONDS_PER_MINUTE * MINUTES_PER_HOUR;

    public static void main(String[] args) {
        LocalDateTime toDateTime = LocalDateTime.of(2014, 9, 9, 19, 46, 45);
        LocalDateTime fromDateTime = LocalDateTime.of(1984, 12, 16, 7, 45, 55);

        Period period = getPeriod(fromDateTime, toDateTime);
        long time[] = getTime(fromDateTime, toDateTime);

        System.out.println(period.getYears() + " years " + 
                period.getMonths() + " months " + 
                period.getDays() + " days " +
                time[0] + " hours " +
                time[1] + " minutes " +
                time[2] + " seconds.");


    }

    private static Period getPeriod(LocalDateTime dob, LocalDateTime now) {
        return Period.between(dob.toLocalDate(), now.toLocalDate());
    }

    private static long[] getTime(LocalDateTime dob, LocalDateTime now) {
        LocalDateTime today = LocalDateTime.of(now.getYear(),
                now.getMonthValue(), now.getDayOfMonth(), dob.getHour(), dob.getMinute(), dob.getSecond());
        Duration duration = Duration.between(today, now);

        long seconds = duration.getSeconds();

        long hours = seconds / SECONDS_PER_HOUR;
        long minutes = ((seconds % SECONDS_PER_HOUR) / SECONDS_PER_MINUTE);
        long secs = (seconds % SECONDS_PER_MINUTE);

        return new long[]{hours, minutes, secs};
    }
}

私が得ている出力は次のとおりです。 29 years 8 months 24 days 12 hours 0 minutes 50 seconds . 私はこの結果を確認しました ウェブサイト (値 12/16/1984 07:45:5509/09/2014 19:46:45 ). 次のスクリーンショットは、その出力を示しています。

月の値の後のフィールドが私のコードから間違って来ているのは確かです。何か提案があれば、非常に助かります。

更新情報

別のサイトで検証したところ、得られた結果が違っていました。それがこちらです。 2つの日付の間の期間を計算する (結果: 29年8ヶ月24日12時間0分50秒)。

更新情報

2つのサイトで2つの異なる結果が得られたので、私の計算のアルゴリズムが正当かどうか疑問です。以下の2つを使用した場合 LocalDateTime オブジェクトを作成します。

LocalDateTime toDateTime = LocalDateTime.of(2014, 9, 10, 6, 40, 45);
LocalDateTime fromDateTime = LocalDateTime.of(1984, 12, 16, 7, 45, 55);

すると、出力が来る。 29 years 8 months 25 days -1 hours -5 minutes -10 seconds.

これより リンク であるべきです。 29 years 8 months 24 days 22 hours, 54 minutes and 50 seconds . だから、アルゴリズムは負の数も処理する必要がある。

質問は、どのサイトがどんな結果を出したかではなく、正しいアルゴリズムを知り、正しい結果を得る必要があることに注意してください。

どのように解決するのですか?

残念ながら、時間をまたぐピリオドクラスもないようなので、自力で計算する必要があるかもしれませんね。

幸いなことに、日付と時刻のクラスには、それをある程度単純化するユーティリティ・メソッドがたくさんあります。ここでは、必ずしも最速とは言えませんが、差分を計算する方法を紹介します。

LocalDateTime fromDateTime = LocalDateTime.of(1984, 12, 16, 7, 45, 55);
LocalDateTime toDateTime = LocalDateTime.of(2014, 9, 10, 6, 40, 45);

LocalDateTime tempDateTime = LocalDateTime.from( fromDateTime );

long years = tempDateTime.until( toDateTime, ChronoUnit.YEARS );
tempDateTime = tempDateTime.plusYears( years );

long months = tempDateTime.until( toDateTime, ChronoUnit.MONTHS );
tempDateTime = tempDateTime.plusMonths( months );

long days = tempDateTime.until( toDateTime, ChronoUnit.DAYS );
tempDateTime = tempDateTime.plusDays( days );


long hours = tempDateTime.until( toDateTime, ChronoUnit.HOURS );
tempDateTime = tempDateTime.plusHours( hours );

long minutes = tempDateTime.until( toDateTime, ChronoUnit.MINUTES );
tempDateTime = tempDateTime.plusMinutes( minutes );

long seconds = tempDateTime.until( toDateTime, ChronoUnit.SECONDS );

System.out.println( years + " years " + 
        months + " months " + 
        days + " days " +
        hours + " hours " +
        minutes + " minutes " +
        seconds + " seconds.");

//prints: 29 years 8 months 24 days 22 hours 54 minutes 50 seconds.

基本的な考え方はこうです:一時的な開始日を作成し、終了までの全期間を取得します。次に、開始日が終了日から1年未満になるように、その日付を年数で調整します。これを各時間単位で降順に繰り返す。

最後に免責事項 : また、サマータイムやその他の暦の変化(サモアのタイムゾーンの変化など)がこの計算にどのように影響するかについては、テストやチェックをしていません。ですから、注意して使用してください。