与其他回答者试图暗示的相反,计算两个日期之间的差异在标准 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”虽然需要一些工作,但这不是问题所在。