Java以这种格式减去两个日期[重复]

Posted

技术标签:

【中文标题】Java以这种格式减去两个日期[重复]【英文标题】:Java subtract two dates in this format [duplicate] 【发布时间】:2012-10-13 18:23:54 【问题描述】:

可能重复:Calculating difference in dates in JavaHow do you subtract Dates in 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 秒

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

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

【问题讨论】:

以上重复仅与***.com/questions/4216745/…结合有效,以便首先将String转换为Date。您可能还想看看 Joda Time:***.com/questions/2179644/… 使用Joda Time 【参考方案1】:

与其他回答者试图暗示的相反,计算两个日期之间的差异在标准 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();
// ...

但是这种天真的方法没有考虑闰年,更不用说夏令时和local-specific changes in 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”需要一些工作,但这不是问题。

【讨论】:

这正是我所需要的,效果很好。但这是一个小问题。如果一年大于 10 000,由于 public static int elapsed(Calendar before, Calendar after, int field); 函数中的循环,它会影响计算速度。我试图在不循环的情况下从目标中减去当前年份,结果是不可预测的。我可以做些什么来处理大于 10 000 的年份而不减慢处理速度?【参考方案2】:

需要使用SimpleDateFormat解析String并创建Date

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

这是SimpleDateFormat的javadoc

【讨论】:

【参考方案3】:

试试这个:

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

【讨论】:

以上是关于Java以这种格式减去两个日期[重复]的主要内容,如果未能解决你的问题,请参考以下文章

以纪元毫秒为单位从当前日期减去两天 java [重复]

PHP:以YYYY-MM-DD格式获取两个日期之间的天数[重复]

以这种格式显示时间和日期[重复]

从日期中添加或减去天数时如何避免周末或节假日[重复]

减去日期时间以获得以月为单位的增量[重复]

C# - 获取两个日期然后只减去年份[重复]