在 Java 中计算两个小时之间的持续时间

Calculate duration between two hours in Java

我想计算 Java 中两个小时 (HH:mm:ss) 之间的时差(持续时间)。在这里,我已经阅读了几个关于这个主题的主题,但我的问题有点不同。

我也无法使用 Joda-Time

示例:

input values:    12:03:00 
                 00:00:00

expected output: 11:57:00

代码:

public static void main(String[] args) throws ParseException {

    Scanner sc = new Scanner(System.in);
    String startTime = sc.next();
    String endTime = sc.next();

    SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");
    Date d1 = sdf.parse(startTime);
    Date d2 = sdf.parse(endTime);
    long elapsed = d2.getTime() - d1.getTime();

    String hms = String.format("%02d:%02d:%02d",
        TimeUnit.MILLISECONDS.toHours(elapsed),
        TimeUnit.MILLISECONDS.toMinutes(elapsed) - TimeUnit.HOURS.toMinutes(TimeUnit.MILLISECONDS.toHours(elapsed)),
        TimeUnit.MILLISECONDS.toSeconds(elapsed) - TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(elapsed)));

    if (elapsed > 0) {
        System.out.println(hms);
    } else {

        elapsed = elapsed * (-1); //otherwise, print hours with '-'

        String hms1 = String.format("%02d:%02d:%02d",
            TimeUnit.MILLISECONDS.toHours(elapsed),
            TimeUnit.MILLISECONDS.toMinutes(elapsed) - TimeUnit.HOURS.toMinutes(TimeUnit.MILLISECONDS.toHours(elapsed)),
            TimeUnit.MILLISECONDS.toSeconds(elapsed) - TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(elapsed)));

        System.out.println(hms1);
    }
}

结果:

expected output: 11:57:00

actual output:   12:03:00 //that's the problem

这是你想做的吗:

import java.time.LocalDateTime;
import java.time.temporal.ChronoUnit;

public class HourTest {
    public static void main(String[] args) throws Exception {
        LocalDateTime ldt1 = LocalDateTime.parse("2015-05-04T12:07:00");
        LocalDateTime ldt2 = LocalDateTime.parse("2015-05-04T00:00:00");

        long seconds = Math.abs(ChronoUnit.SECONDS.between(ldt1, ldt2));

        String hms = String.format("%02d:%02d:%02d", seconds / 3600, (seconds / 60) % 60, seconds % 60);

        // Prints 12:07:00. I tried it.
        System.out.println(hms);
    }
}

这里的基本缺陷是你想要两个时间之间的最近距离。当您构建日期对象时,即使您只格式化 Hour:Minute:Second 它仍然存储 day/month/year 等...对于日期 12:03:00 和 00:00:00 它默认它们为同一天,所以与(午夜到中午)的区别是你第二天(中午到午夜)没有得到的。您的解决方案是检查时间是否小于 12(军用时间),如果小于 12,则在当天加 1。

操作方法如下:

    String t1 = "12:03:00";
    String t2 = "00:00:00";
    SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");

    Date d1 = sdf.parse(t1);
    Date d2 = sdf.parse(t2);
    Calendar c1 = Calendar.getInstance();
    Calendar c2 = Calendar.getInstance();
    c1.setTime(d1);
    c2.setTime(d2);

    if(c2.get(Calendar.HOUR_OF_DAY) < 12) {
        c2.set(Calendar.DAY_OF_YEAR, c2.get(Calendar.DAY_OF_YEAR) + 1);
    }
    long elapsed = c2.getTimeInMillis() - c1.getTimeInMillis();

tl;博士

Duration.between( 
    LocalTime.parse( "12:03:00" ) , 
    LocalTime.parse( "23:59:59.999999999" ) 
).plusNanos( 1 ).withNanos( 0 ) 

PT11H57M

使用纳秒

现代方法使用 java.time classes.

要注意的是,只有一天中的某个时间,没有午夜。因此,您显然打算在一天结束时使用的 00:00 实际上被解释为一天开始。

一天结束前只有最后一纳秒。 LocalTime 为最后一个纳秒定义了一个常量:LocalTime.MAX = 23:59:59.999999999

由于您只关心整秒,我们可以利用小数秒。如果您的结束时间恰好是 00:00:00,请替换为 LocalTime.MAX。然后计算一个Duration对象。您可以添加一个纳秒,然后通过将小数秒(纳秒)设置为零来截断生成的小数秒。

对于 00:00:00 以外的结束时间,数学仍然有效。添加纳秒会得到 ….000000001 秒的分数,Duration::withNanos 会截断不需要的分数。

// This code assumes the inputs are *always* in whole seconds, without any fraction-of-second.

LocalTime start = LocalTime.parse( "12:03:00" );
LocalTime stop = LocalTime.parse( "00:00:00" );

if( stop.equals( LocalTime.MIN ) ) {
    stop = LocalTime.MAX ; // `23:59:59.999999999`
}

Duration d = Duration.between( start , stop );
d = d.plusNanos( 1 ).withNanos( 0 ) ;

System.out.println( "start/stop: " + start + "/" + stop );
System.out.println( "d: " + d );

看到这个 code run live at IdeOne.com

PT11H57M

正在格式化

toString 的输出是标准的 ISO 8601 formatDuration class 可以解析并生成这些字符串。

我强烈建议不要以时间形式表示时间跨度,HH:MM:SS。这是模棱两可的,在人们阅读时经常会造成混淆。

但如果您坚持使用该格式,则必须自己构建字符串。 Duration class 缺少在其他 java.time class 中看到的 format 方法。奇怪的是,这个 class 本来就没有提取部分的方法,天数,小时数,分钟数,秒数,小数秒。请参阅 this Question 进行讨论。 Java9 带来这样的方法,命名为to…Part.

在使用 Java 8 时,我建议对 ISO 8601 格式的输出进行字符串操作。将 PT 替换为空字符串。如果有 M,请将 H 替换为冒号。如果有 S,请将 M 替换为冒号,并将 S 替换为空字符串。如果没有 S,请将 M 替换为空字符串。我相信您可以在 Stack Overflow 上找到此代码。

您可以使用 java.time.Duration which is modelled on ISO-8601 standards and was introduced with Java-8 as part of JSR-310 implementation。随着 Java-9 引入了一些更方便的方法。

演示:

import java.time.Duration;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;

public class Main {
    public static void main(String[] args) {
        LocalDate today = LocalDate.now();
        LocalTime startTime = LocalTime.parse("12:03:00");
        LocalTime endTime = LocalTime.parse("00:00:00");

        LocalDateTime startDateTime = today.atTime(startTime);
        LocalDateTime endDateTime = today.atTime(endTime);

        if (startDateTime.isAfter(endDateTime)) {
            endDateTime = endDateTime.with(LocalTime.MIN).plusDays(1).with(endTime);
        }

        Duration duration = Duration.between(startDateTime, endDateTime);
        // Default format
        System.out.println(duration);

        // Custom format
        // ####################################Java-8####################################
        String formattedDuration = String.format("%02d:%02d:%02d", duration.toHours(), duration.toMinutes() % 60,
                duration.toSeconds() % 60);
        System.out.println(formattedDuration);
        // ##############################################################################

        // ####################################Java-9####################################
        formattedDuration = String.format("%02d:%02d:%02d", duration.toHoursPart(), duration.toMinutesPart(),
                duration.toSecondsPart());
        System.out.println(formattedDuration);
        // ##############################################################################
    }
}

输出:

PT11H57M
11:57:00
11:57:00

Trail: Date Time.

了解现代日期时间 API
  • 出于任何原因,如果您必须坚持使用 Java 6 或 Java 7,您可以使用 ThreeTen-Backport,它向后移植了大部分 java.time Java 6 和 7 的功能。
  • 如果您正在为 Android 项目工作,并且您的 Android API 水平仍然不符合 Java-8,请检查 Java 8+ APIs available through desugaring and