String2date计算两个date 间相差的天数(java)

测试如下:

public long date2Date(String sdate1,String sdate2) throws Exception{
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date date1 = sdf.parse(sdate1);
Date date2 = sdf.parse(sdate2);

return date2.getTime() / (24*60*60*1000) - date1.getTime() / (24*60*60*1000);
}

@Test
public void testDate2Date() throws Exception{
long date = date2Date("2012-4-11","2012-4-23");
System.out.println("相差:"+date);

}

原来以为,求两个时间之日期差应该不是问题,直到有一天...。 我曾经辅导过的一位开发员向我求教,我还不以为很难,尝试用Calendar/GregorianCalendar类来计算,看起来代码太复杂,放弃。苦想半小时,最后想出一句话,写出来在MSN上发给他。当时的感觉路子很野,上不了台面。这句话也是下面要介绍的简洁方法。
再后来,我的一位前同事发来邮件,说起网上一大牛发布的blog,是一个牛人数落Java标准库的,挺有意思。需求很简单:有两个Date型的变量
(a, b),如何计算两者相隔的天数?你有简单的解决方案吗?(据这个牛人说,还是很麻烦的哦)
看了大牛的方法,暗地思忖:既然Java标准库不能解决,我的方法不是很简单吗?!我的底气腾腾往上涨。

====
我的方法:

public long differ(Datedate1,Datedate2)
... {
//returndate1.getTime()/(24*60*60*1000)-date2.getTime()/(24*60*60*1000);
returndate2.getTime()/86400000-date1.getTime()/86400000;//用立即数,减少乘法计算的开销
}

问题分析:求日期差,重要的是时间对象的日期部分,与时分秒没有关系。所以问题的关键是过滤掉时分秒,保留日期部分。干的活象低通滤波器,滤掉高频杂波,保留低频信号。
解决思路:一天的秒数是24*60*60*1000=86400000=Senconds。时间对象date1.getTime()是自1970年1月1日零点以来的秒数,用它整除以Seconds
,就得到自1970年1月1日以来的天数。注意这里用的是整除符号,时分秒被过滤掉了。对date2也施行同样的操作,得到date2的天数。date2的天数减去date1的天数,所得结果就是日期差。
举例来说,用最简单的例子,假设date1等于1970年1月5日15点整,date2等于1970年1月6日14点整。date1整除以seconds的结果是4,date2整除以
seconds的结果是5,date2与date1相差5-4=1天,与我们的预期相符。

====
我的同事Kun给出了方法:获取二者的milliseconds相减,然后/24/60/60。
这个方法应该是不正确的。因为秒数相减,得到的是二者的时间差,并没有滤掉时分秒。
计算结果:用刚才的例子验算一下,date2.getTime() - date1.getTime() 等于相当于23小时的秒数,再整除以seconds,结果是0天,与预期的1
天不相符。
Kun的方法与正确方法的区别在于:先做减法,还是先做整除法。


====
同事在网上找到的方法:

SimpleDateFormatmyFormatter = new SimpleDateFormat( " yyyy-MM-dd " );
java.util.Datedate
= myFormatter.parse( " 2003-05-1 " );
java.util.Datemydate
= myFormatter.parse( " 1899-12-30 " );
long day = (date.getTime() - mydate.getTime()) / ( 24 * 60 * 60 * 1000 );
out.println(day);


结果应该是正确的,它根本就没有时分秒,有了时分秒与Kun的方法等价。不过用起来会很麻烦,因为要先得到字符串的日期,才能用它。

====
最后看看大牛的方法:
大客地址:
http://blog.csdn.net/rmartin/archive/2006/12/22/1452867.aspx

private int daysBetween(Datenow,DatereturnDate) ... {
CalendarcNow
=Calendar.getInstance();
CalendarcReturnDate
=Calendar.getInstance();
cNow.setTime(now);
cReturnDate.setTime(returnDate);
setTimeToMidnight(cNow);
setTimeToMidnight(cReturnDate);
longtodayMs=cNow.getTimeInMillis();
longreturnMs=cReturnDate.getTimeInMillis();
longintervalMs=todayMs-returnMs;
returnmillisecondsToDays(intervalMs);
}


private int millisecondsToDays( long intervalMs) ... {
return(int)(intervalMs/(1000*86400));
}


private void setTimeToMidnight(Calendarcalendar) ... {
calendar.set(Calendar.HOUR_OF_DAY,
0);
calendar.set(Calendar.MINUTE,
0);
calendar.set(Calendar.SECOND,
0);
}

看起来很复杂吧,没有求证过,应该是正确的吧!

  • 0
    点赞
  • 0
    评论
  • 0
    收藏
  • 扫一扫,分享海报

©️2022 CSDN 皮肤主题:大白 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值