java:通过Calendar类正确计算两日期之间的间隔
在开发Android应用时偶然需要用到一个提示用户已用天数的功能,从实现上来看无非就是持久化存入用户第一次使用应用的时间firstTime(通过SharedPreferences 、xml、sqlite等),当用户再次使用应用时取得此时时间presentTime,通过两个时间计算其间隔天数。
当取得两个时间变量后,网上计算日期间隔的做法通常是这样的(获得两时间的毫秒数之差再进行处理):
- long beginTime = beginDate.getTime();
- long endTime = endDate.getTime();
- 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:
- 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
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:
- Calendar beginCalendar = Calendar.getInstance();
- beginCalendar.set(2017,2,2,20,20,20);<span style="white-space: pre;"> </span>//设定时间为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
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):
所以上述代码需要改进,使其能够识别跨天这种特殊情况:
- 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; //不相等说明确实未跨天
- }
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时间值是否相等,从而确定返回值
而没有直接
- 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类正确计算两日期之间的间隔的更多相关文章
- C++练习 | 计算两日期之间天数差
#include<iostream> #include<string> #include<cstring> using namespace std; class D ...
- MySql计算两日期时间之间相差的天数,秒数,分钟数,周数,小时数
MySql计算两日期时间之间相差的天数,秒数,分钟数,周数,小时数 计算两日期时间之间相差的天数,秒数,分钟数,周数,小时数,这里主要分享的是通过MySql内置的函数 TimeStampDiff() ...
- Java基础-Calendar类常用方法介绍
Java基础-Calendar类常用方法介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Calendar类概念 Calendar 类是一个抽象类,它为特定瞬间与一组诸如 Y ...
- Java学习--Calendar 类的应用
Calendar 类的应用 Date 类最主要的作用就是获得当前时间,同时这个类里面也具有设置时间以及一些其他的功能,但是由于本身设计的问题,这些方法却遭到众多批评,不建议使用,更推荐使用 Calen ...
- JAVA中Calendar 类的应用
转自:https://www.imooc.com/code/2340 侵删! Date 类最主要的作用就是获得当前时间,同时这个类里面也具有设置时间以及一些其他的功能,但是由于本身设计的问题,这些方法 ...
- Java.util.Calendar类
Java.util.Calendar类 package myProject; import java.text.SimpleDateFormat; import java.util.Calendar; ...
- 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 ...
- java 使用Date类、Calendar类,实现增加日期
Date date= new Date(); System.out.println("Date date= new Date()中的date: "+date); 输出对象date: ...
- Java的Calendar类
通过Date类我们可以创建并格式化一个日期对象,但是如何才能设置和获取日期数据的特定部分呢?----Calendar类 Calendar类是一个抽象类,在实际使用时实现特定的子类的对象,通过getIn ...
随机推荐
- Redis持久化配置-AOF
redis的持久化有rdb和aof两种. rdb是记录一段时间内的操作,一般的配置是一段时间内操作超过多少次就持久化.aof可以实现每次操作都持久化. 这里我们使用aof. 配置方式,打开redis的 ...
- 业务ID 生成规则
在实际业务中,是否碰到过这种场景: 我们需要一个单号,要在业务系统里面保证唯一,保证增长? 在运营过程,需要知道业务单发生的时间,最好不用经过系统查找就知道发生的时间? 在排障过程中,不用再次查找就知 ...
- Oracle数据库 中的基础的一些语法结构
方括号里的内容为可选项 大括号是必填 1PL/SQL结构块 DECLARE /* * 声明部分——定义常量.变量.复杂数据类型.游标.用户自定义异常 */ BEGIN /* * 执行部分——PL/SQ ...
- Java 和 JSP 实现网站访问量统计 (刷新过滤)
java 和 JSP 实现的统计网站访问量,不需要数据库,将数据存储在 指定位置的 txt 文件中,代码块分为两部分 首先, java 部分: import java.io.File; import ...
- Prim算法和Kruskal算法求最小生成树
Prim算法 连通分量是指图的一个子图,子图中任意两个顶点之间都是可达的.最小生成树是连通图的一个连通分量,且所有边的权值和最小. 最小生成树中,一个顶点最多与两个顶点邻接:若连通图有n个顶点,则最小 ...
- Linux skbuff注释笔记
SKB结构定义 /usr/src/linux/include/linux/skbuff.h sk_buff_head: struct sk_buff_head { //SKB的头结点 /* The ...
- bootstrap栅格系统的属性及使用
栅格系统 媒体查询 在栅格系统中,我们在 Less 文件中使用以下媒体查询(media query)来创建关键的分界点阈值. 小屏幕(平板,大于等于 768px) @media (min-width: ...
- PowerDesigner设置默认值名称规则
一.需求背景: 使用PowerDesigner创建表时,若设置某列默认值时,自动生成规则的默认值名称.比如说:DF_表名_列名 二.设置步骤: 1.选择Database—>Edit Curren ...
- 使用iTextSharp 解析html生成pdf,xmlworker不支持中文的解决办法
http://www.micmiu.com/opensource/expdoc/itext-xml-worker-cn/ 参考上面的文章,虽然是java的,但是和.net是对应的. 下载 html ...
- mybatis在mysql中的分页扩展
applicationContext.xml <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlS ...