在开发Android应用时偶然需要用到一个提示用户已用天数的功能,从实现上来看无非就是持久化存入用户第一次使用应用的时间firstTime(通过SharedPreferences 、xml、sqlite等),当用户再次使用应用时取得此时时间presentTime,通过两个时间计算其间隔天数。

当取得两个时间变量后,网上计算日期间隔的做法通常是这样的(获得两时间的毫秒数之差再进行处理):

  1. long beginTime = beginDate.getTime();
  2. long endTime = endDate.getTime();
  3. long betweenDays = (long)((endTime - beginTime) / (1000 * 60 * 60 *24));
long beginTime = beginDate.getTime();
long endTime = endDate.getTime();
long betweenDays = (long)((endTime - beginTime) / (1000 * 60 * 60 *24));

下面测试一下两种特殊情况:

情况1:

  1. Calendar beginCalendar = Calendar.getInstance();
  2. beginCalendar.set(2017,2,3,1,0,0);      //设定时间为2017年3月3日 1:0:0
  3. Calendar endCalendar = Calendar.getInstance();
  4. endCalendar.set(2017,2,3,14,0,0);       //设定时间为2017年3月3日 14:0:0
  5. long beginTime = beginCalendar.getTime().getTime();
  6. long endTime = endCalendar.getTime().getTime();
  7. long betweenDays = (long)((endTime - beginTime) / (1000 * 60 * 60 *24));
  8. System.out.println(betweenDays);        //输出为0
Calendar beginCalendar = Calendar.getInstance();
beginCalendar.set(2017,2,3,1,0,0); //设定时间为2017年3月3日 1:0:0
Calendar endCalendar = Calendar.getInstance();
endCalendar.set(2017,2,3,14,0,0); //设定时间为2017年3月3日 14:0:0
long beginTime = beginCalendar.getTime().getTime();
long endTime = endCalendar.getTime().getTime();
long betweenDays = (long)((endTime - beginTime) / (1000 * 60 * 60 *24));
System.out.println(betweenDays); //输出为0

情况2:

  1. Calendar beginCalendar = Calendar.getInstance();
  2. beginCalendar.set(2017,2,2,20,20,20);<span style="white-space: pre;">       </span>//设定时间为2017年3月2日20:20:20
  3. Calendar endCalendar = Calendar.getInstance();
  4. endCalendar.set(2017,2,3,10,10,10);     //设定时间为2017年3月3日10:10:10
  5. long beginTime = beginCalendar.getTime().getTime();
  6. long endTime = endCalendar.getTime().getTime();
  7. long betweenDays = (long)((endTime - beginTime) / (1000 * 60 * 60 *24));
  8. System.out.println(betweenDays);        //输出为0,但其实应该为1
Calendar beginCalendar = Calendar.getInstance();
beginCalendar.set(2017,2,2,20,20,20); //设定时间为2017年3月2日20:20:20
Calendar endCalendar = Calendar.getInstance();
endCalendar.set(2017,2,3,10,10,10); //设定时间为2017年3月3日10:10:10
long beginTime = beginCalendar.getTime().getTime();
long endTime = endCalendar.getTime().getTime();
long betweenDays = (long)((endTime - beginTime) / (1000 * 60 * 60 *24));
System.out.println(betweenDays); //输出为0,但其实应该为1

可见,对于本应该时间间隔为1的情况该代码输出为0。究其原因,是因为当 两日期的毫秒数之差 < 一天的毫秒数(1000*60*60*24)时,即使两日期跨了一天,强制类型转换后结果也会变成0。通过下图可以看出毫秒数之差小于一天的两种情况。

情况1(毫秒数之差小于一天且不跨天,日期间隔应为0):

情况2(毫秒数之差小于一天且跨天,日期间隔应为1):

所以上述代码需要改进,使其能够识别跨天这种特殊情况:

  1. public static int getTimeDistance(Date beginDate , Date endDate ) {
  2. Calendar beginCalendar = Calendar.getInstance();
  3. beginCalendar.setTime(beginDate);
  4. Calendar endCalendar = Calendar.getInstance();
  5. endCalendar.setTime(endDate);
  6. long beginTime = beginCalendar.getTime().getTime();
  7. long endTime = endCalendar.getTime().getTime();
  8. int betweenDays = (int)((endTime - beginTime) / (1000 * 60 * 60 *24));//先算出两时间的毫秒数之差大于一天的天数
  9. endCalendar.add(Calendar.DAY_OF_MONTH, -betweenDays);//使endCalendar减去这些天数,将问题转换为两时间的毫秒数之差不足一天的情况
  10. endCalendar.add(Calendar.DAY_OF_MONTH, -1);//再使endCalendar减去1天
  11. if(beginCalendar.get(Calendar.DAY_OF_MONTH)==endCalendar.get(Calendar.DAY_OF_MONTH))//比较两日期的DAY_OF_MONTH是否相等
  12. return betweenDays + 1; //相等说明确实跨天了
  13. else
  14. return betweenDays + 0; //不相等说明确实未跨天
  15. }
public static int getTimeDistance(Date beginDate , Date endDate ) {
Calendar beginCalendar = Calendar.getInstance();
beginCalendar.setTime(beginDate);
Calendar endCalendar = Calendar.getInstance();
endCalendar.setTime(endDate);
long beginTime = beginCalendar.getTime().getTime();
long endTime = endCalendar.getTime().getTime();
int betweenDays = (int)((endTime - beginTime) / (1000 * 60 * 60 *24));//先算出两时间的毫秒数之差大于一天的天数 endCalendar.add(Calendar.DAY_OF_MONTH, -betweenDays);//使endCalendar减去这些天数,将问题转换为两时间的毫秒数之差不足一天的情况
endCalendar.add(Calendar.DAY_OF_MONTH, -1);//再使endCalendar减去1天
if(beginCalendar.get(Calendar.DAY_OF_MONTH)==endCalendar.get(Calendar.DAY_OF_MONTH))//比较两日期的DAY_OF_MONTH是否相等
return betweenDays + 1; //相等说明确实跨天了
else
return betweenDays + 0; //不相等说明确实未跨天
}

代码中有个细节需要注意,我先使 endCalendar.add(Calendar.DAY_OF_MONTH, -1),然后比较两日期的DAY_OF_MONTH时间值是否相等,从而确定返回值

而没有直接

  1. return betweenDays + endCalendar.get(Calendar.DAY_OF_MONTH)-beginCalendar.get(Calendar.DAY_OF_MONTH);
return betweenDays + endCalendar.get(Calendar.DAY_OF_MONTH)-beginCalendar.get(Calendar.DAY_OF_MONTH);

是因为有可能出现跨月的情况,例如beginCalendar的日期为2月28日,endCalendar日期为3月1日,endCalendar.get(Calendar.DAY_OF_MONTH)-beginCalendar.get(Calendar.DAY_OF_MONTH)的值就会是-27而不是预期中的1。而使用endCalendar.add(Calendar.DAY_OF_MONTH, -1),Calendar类的方法会自动的对跨月情况进行转换。

java:通过Calendar类正确计算两日期之间的间隔的更多相关文章

  1. C++练习 | 计算两日期之间天数差

    #include<iostream> #include<string> #include<cstring> using namespace std; class D ...

  2. MySql计算两日期时间之间相差的天数,秒数,分钟数,周数,小时数

    MySql计算两日期时间之间相差的天数,秒数,分钟数,周数,小时数 计算两日期时间之间相差的天数,秒数,分钟数,周数,小时数,这里主要分享的是通过MySql内置的函数 TimeStampDiff() ...

  3. Java基础-Calendar类常用方法介绍

    Java基础-Calendar类常用方法介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Calendar类概念 Calendar 类是一个抽象类,它为特定瞬间与一组诸如 Y ...

  4. Java学习--Calendar 类的应用

    Calendar 类的应用 Date 类最主要的作用就是获得当前时间,同时这个类里面也具有设置时间以及一些其他的功能,但是由于本身设计的问题,这些方法却遭到众多批评,不建议使用,更推荐使用 Calen ...

  5. JAVA中Calendar 类的应用

    转自:https://www.imooc.com/code/2340 侵删! Date 类最主要的作用就是获得当前时间,同时这个类里面也具有设置时间以及一些其他的功能,但是由于本身设计的问题,这些方法 ...

  6. Java.util.Calendar类

    Java.util.Calendar类 package myProject; import java.text.SimpleDateFormat; import java.util.Calendar; ...

  7. HDOJ(HDU) 2133 What day is it(认识下Java的Calendar类---日期类)

    Problem Description Today is Saturday, 17th Nov,2007. Now, if i tell you a date, can you tell me wha ...

  8. java 使用Date类、Calendar类,实现增加日期

    Date date= new Date(); System.out.println("Date date= new Date()中的date: "+date); 输出对象date: ...

  9. Java的Calendar类

    通过Date类我们可以创建并格式化一个日期对象,但是如何才能设置和获取日期数据的特定部分呢?----Calendar类 Calendar类是一个抽象类,在实际使用时实现特定的子类的对象,通过getIn ...

随机推荐

  1. PL/SQL程序设计

    1 PL/SQL简介 1 什么是PL/SQL? PL/SQL是 Procedure Language & Structured Query Language 的缩写.PL/SQL是对SQL语言 ...

  2. gulp4.0 前端构建脚手架

    最近看了下gulp4.0的升级,感觉和3.0相比变化还是比较大的,很多3.0的写法和插件会出现一些莫名其妙的变化,详细的变化就先不说了,这里我直接把我配置好的代码拿过来吧,方便各位可以更好的学习和使用 ...

  3. 自然语言处理--Word2vec(二)

    前一篇,word2vec(一)主要讲了word2vec一些表层概念,以及主要介绍CBOW方法来求解词向量模型,这里主要讲论文 Distributed Representations of Words ...

  4. 梯度下降法的三种形式-BGD、SGD、MBGD

    在应用机器学习算法时,我们通常采用梯度下降法来对采用的算法进行训练.其实,常用的梯度下降法还具体包含有三种不同的形式,它们也各自有着不同的优缺点. 下面我们以线性回归算法来对三种梯度下降法进行比较. ...

  5. alloca() 是什么?为什么不提倡使用它?

    在调用 alloca() 的函数返回的时候, 它分配的内存会自动释放.也就是说, 用 alloca 分配的内存在某种程度上局部于函数的 ``堆栈帧"  或上下文中. alloca() 不具可 ...

  6. TCP/IP协议栈概述及各层包头分析

    TCP/IP协议栈中各层包头的分析 Protocol列表示的是该数据包最高层对应的协议,Length列表示该包的长度(包括从底层的协议到最高层的协议,其中包头一般是,链路层14字节,IP20字节,TC ...

  7. 【LeetCode题解】21_合并两个有序链表

    目录 21_合并两个有序链表 描述 解法一:迭代 思路 Java 实现 Python 实现 解法二:递归 思路 Java 实现 Python 实现 21_合并两个有序链表 描述 将两个有序链表合并为一 ...

  8. C语言----<另类>神奇的"Hello World!"

    先上代码 #include <iostream> using namespace std; void a() { printf("Hello World!"); } v ...

  9. AspxGridView中行的双击事件

    ClientSideEvents-RowDblClick="clike" function clike(s, e) {           grdList.GetRowKey(e. ...

  10. jquery 关于使用 append 追加 元素后 事件无法触发

    当在使用js或jQuery创建元素时,用 on(事件,function(){代码}) 或者 事件(function(){代码 })绑定事件时 在使用append添加元素后 由于是在页面加载完成之后进行 ...