5

可能重复:
在 Java 中计算日期的差异
如何在 Java 中减去日期?

我正在从一个字符串中解析两个日期,如下所示:

Oct 15, 2012 1:07:13 PM
Oct 23, 2012 03:43:34 PM

我需要做的是找出这两个日期之间的区别,例如:

Oct 23, 2012 03:43:34 PM - Oct 15, 2012 1:07:13 PM

= 8 天 2 小时 36 分 21 秒

^ 这是我需要得到的两个日期/时间

我相信我需要解析格式并将其转换为另一种格式,然后减去两者之间的差异并进行数学运算以获得之间的天/小时/分钟/秒

4

3 回答 3

12

与其他回答者试图暗示的相反,计算两个日期之间的差异在标准 Java SE 中并不是那么简单。

您的第一步确实是将这些字符串转换为可用Date实例。您可以使用SimpleDateFormat. 这是一个启动示例:

String string1 = "Oct 15, 2012 1:07:13 PM";
String string2 = "Oct 23, 2012 03:43:34 PM";

SimpleDateFormat sdf = new SimpleDateFormat("MMM d, yyyy h:mm:ss a", Locale.ENGLISH);

Date date1 = sdf.parse(string1);
Date date2 = sdf.parse(string2);

(请注意此处可选参数的重要性Locale,这在有关将字符串转换为日期的答案中经常被忽略)

您的下一步是计算这两个日期之间的差异。当您受限于标准 Java SE API 时,这是一项糟糕的工作。你能得到的最好的是java.util.Calendar.


请注意,您当然可以减去毫秒并使用通常的算术运算符计算差异。

long differenceInMillis = date2.getTime() - date1.getTime();
// ...

但是这种天真的方法没有考虑闰年,更不用说夏令时和本地特定的 datetime 变化


至于java.util.Calendar方法,您基本上需要Calendar#add()在计数器循环中使用来获取年、月和日的经过值。这需要适当考虑闰年、夏令时和当地特定的时间干扰。

首先创建这个辅助方法来消除一些样板代码:

public static int elapsed(Calendar before, Calendar after, int field) {
    Calendar clone = (Calendar) before.clone(); // Otherwise changes are been reflected.
    int elapsed = -1;
    while (!clone.after(after)) {
        clone.add(field, 1);
        elapsed++;
    }
    return elapsed;
}

现在您可以按如下方式计算经过的时间:

Calendar start = Calendar.getInstance();
start.setTime(date1);
Calendar end = Calendar.getInstance();
end.setTime(date2);

Integer[] elapsed = new Integer[6];
Calendar clone = (Calendar) start.clone(); // Otherwise changes are been reflected.
elapsed[0] = elapsed(clone, end, Calendar.YEAR);
clone.add(Calendar.YEAR, elapsed[0]);
elapsed[1] = elapsed(clone, end, Calendar.MONTH);
clone.add(Calendar.MONTH, elapsed[1]);
elapsed[2] = elapsed(clone, end, Calendar.DATE);
clone.add(Calendar.DATE, elapsed[2]);
elapsed[3] = (int) (end.getTimeInMillis() - clone.getTimeInMillis()) / 3600000;
clone.add(Calendar.HOUR, elapsed[3]);
elapsed[4] = (int) (end.getTimeInMillis() - clone.getTimeInMillis()) / 60000;
clone.add(Calendar.MINUTE, elapsed[4]);
elapsed[5] = (int) (end.getTimeInMillis() - clone.getTimeInMillis()) / 1000;

System.out.format("%d years, %d months, %d days, %d hours, %d minutes, %d seconds", elapsed);

很丑,是的。

如果您经常在 Java 中使用日期和时间,那么您可能会发现Joda time是 walhalla。这是一个具体的启动示例,说明如何使用纯 Joda Time 完成所有操作:

String string1 = "Oct 15, 2012 1:07:13 PM";
String string2 = "Oct 23, 2012 03:43:34 PM";

DateTimeFormatter dtf = DateTimeFormat.forPattern("MMM d, yyyy h:mm:ss a").withLocale(Locale.ENGLISH);

DateTime dateTime1 = dtf.parseDateTime(string1);
DateTime dateTime2 = dtf.parseDateTime(string2);
Period period = new Period(dateTime1, dateTime2);

PeriodFormatter formatter = new PeriodFormatterBuilder()
    .appendYears().appendSuffix(" years ")
    .appendMonths().appendSuffix(" months ")
    .appendWeeks().appendSuffix(" weeks ")
    .appendDays().appendSuffix(" days ")
    .appendHours().appendSuffix(" hours ")
    .appendMinutes().appendSuffix(" minutes ")
    .appendSeconds().appendSuffix(" seconds ")
    .printZeroNever()
    .toFormatter();

String elapsed = formatter.print(period);
System.out.println(elapsed);

好多了,对吧?复数“s”虽然需要一些工作,但这不是问题所在。

于 2012-10-23T19:17:00.523 回答
0

您需要使用SimpleDateFormat来解析字符串并创建Date

然后你可以找到日期之间的差异。

这是SimpleDateFormat的 javadoc

于 2012-10-23T18:32:28.277 回答
-4

试试这个:

Calendar ca1 = Calendar.getInstance();
ca1.set(2012,05,25);
// Addition of date in java        
ca1.add(Calendar.DATE, 23); // Add 23 days in Dates in Calendar
ca1.add(Calendar.MONTH, 2); // Add 2 Month in Date in Calendar
ca1.add(Calendar.YEAR, 4); // add 4 Year in Date in Calendar

ca1.add(Calendar.DATE, -23); // sub 23 days in Dates in Calendar
ca1.add(Calendar.MONTH, -2); // sub 2 Month in Date in Calendar
ca1.add(Calendar.YEAR, -4); // sub 4 Year in Date in Calendar
于 2012-10-23T18:35:48.283 回答