Calendar类经常用法 日期间的转换 set方法有巨坑
2017年3月31日,之所以说今天的日期,就是跟bug有关,且看以下代码
calendarInstance.set(Calendar.MONTH, calendarInstance.get(Calendar.MONTH) + 1 );
这句代码意思非常明确。获取当前日期的下一月。
正常来说,比方不是今天,是3月30日,这句代码运行一点问题没有,得到4月30日,可是今天是3月31日,得到的日期是4月31日,而4月没有31日,顺延到5月1日,那么,这个不定时出现的bug就出现了。还非常easy迷惑你,由于第二天bug自己主动恢复了。一点问题没有。摊手。。
。
正确使用方法应该是以下,得到4月30日
calendarInstance.add(Calendar.MONTH, 1 );
顺便列一下Calendar的使用方法
Calendar类的静态方法getInstance()能够初始化一个日历对象:
Calendar now = Calendar.getInstance();
能够使用以下三个方法把日历定到不论什么一个时间:
set(int year ,int month,int date)
set(int year ,int month,int date,int hour,int minute)
set(int year ,int month,int date,int hour,int minute,int second)
假设想获得年份、月份、小时等信息能够使用:
Now.get(Calendar.Month) 这个方案 0表示一月,1表示二月
get(Calendar.DAY_OF_MONTH)获得这个月的第几天
get(Calendar.DAY_OF_WEEK)获得这个星期的第几天
get(Calendar.DAY_OF_YEAR)获得这个年的第几天
getTimeMillis()获得当前时间的毫秒表示
例如以下是Calendar类方法简单介绍
abstract void add(int field, int amount) 依据日历的规则。为给定的日历字段加入或减去指定的时间量。
boolean after(Object when) 推断此 Calendar 表示的时间是否在指定 Object 表示的时间之后,返回推断结果。
boolean before(Object when) 推断此 Calendar 表示的时间是否在指定 Object 表示的时间之前,返回推断结果。
void clear()将此 Calendar 的所日历字段值和时间值(从历元至如今的毫秒偏移量)设置成没有定义。
void clear(int field) 将此 Calendar 的给定日历字段值和时间值(从历元至如今的毫秒偏移量)设置成没有定义。
Object clone()创建并返回此对象的一个副本。
int compareTo(Calendar anotherCalendar) 比較两个 Calendar 对象表示的时间值(从历元至如今的毫秒偏移量)。
protected void complete()填充日历字段中全部未设置的字段。
protected abstract void computeFields()将当前毫秒时间值 time 转换为 fields[] 中的日历字段值。
protected abstract void computeTime()将 fields[] 中的当前日历字段值转换为毫秒时间值 time。
boolean equals(Object obj) 将此 Calendar 与指定 Object 比較。
int get(int field)返回给定日历字段的值。
int getActualMaximum(int field)给定此 Calendar 的时间值,返回指定日历字段可能拥有的最大值。
int getActualMinimum(int field)给定此 Calendar 的时间值,返回指定日历字段可能拥有的最小值。
static Locale[] getAvailableLocales()返回全部语言环境的数组。此类的 getInstance 方法能够为其返回本地化的实例。
String getDisplayName(int field, int style, Locale locale) 返回给定 style 和 locale 下的日历 field 值的字符串表示形式。
Map<String,Integer> getDisplayNames(int field, int style, Locale locale) 返回给定 style 和 locale 下包括日历 field 全部名称的 Map 及其对应字段值。
int getFirstDayOfWeek()获取一星期的第一天;比如,在美国,这一天是 SUNDAY。而在法国,这一天是 MONDAY。
abstract int getGreatestMinimum(int field)返回此 Calendar 实例给定日历字段的最高的最小值。
static Calendar getInstance() 使用默认时区和语言环境获得一个日历。
static Calendar getInstance(Locale aLocale) 使用默认时区和指定语言环境获得一个日历。
static Calendar getInstance(TimeZone zone) 使用指定时区和默认语言环境获得一个日历。
static Calendar getInstance(TimeZone zone, Locale aLocale) 使用指定时区和语言环境获得一个日历。
abstract int getLeastMaximum(int field) 返回此 Calendar 实例给定日历字段的最低的最大值。
abstract int getMaximum(int field) 返回此 Calendar 实例给定日历字段的最大值。
int getMinimalDaysInFirstWeek()获取一年中第一个星期所需的最少天数。比如。假设定义第一个星期包括一年第一个月的第一天,则此方法将返回 1。
abstract int getMinimum(int field) 返回此 Calendar 实例给定日历字段的最小值。
Date getTime()返回一个表示此 Calendar 时间值(从历元至如今的毫秒偏移量)的 Date 对象。
long getTimeInMillis()返回此 Calendar 的时间值,以毫秒为单位。
TimeZone getTimeZone()获得时区。
int hashCode()返回该此日历的哈希码。
protected int internalGet(int field)返回给定日历字段的值。
boolean isLenient()推断日期/时间的解释是否为宽松的。
boolean isSet(int field) 确定给定日历字段是否已经设置了一个值,当中包含由于调用 get 方法触发内部字段计算而导致已经设置该值的情况。
abstract void roll(int field, boolean up) 在给定的时间字段上加入或减去(上/下)单个时间单元,不更改更大的字段。
void roll(int field, int amount) 向指定日历字段加入指定(有符号的)时间量,不更改更大的字段。
void set(int field, int value) 将给定的日历字段设置为给定值。
void set(int year, int month, int date) 设置日历字段 YEAR、MONTH 和 DAY_OF_MONTH 的值。
void set(int year, int month, int date, int hourOfDay, int minute) 设置日历字段 YEAR、MONTH、DAY_OF_MONTH、HOUR_OF_DAY 和 MINUTE 的值。
void set(int year, int month, int date, int hourOfDay, int minute, int second) 设置字段 YEAR、MONTH、DAY_OF_MONTH、HOUR、MINUTE 和 SECOND 的值。
void setFirstDayOfWeek(int value) 设置一星期的第一天是哪一天;比如,在美国,这一天是 SUNDAY,而在法国。这一天是 MONDAY。
void setLenient(boolean lenient) 指定日期/时间解释是否是宽松的。
void setMinimalDaysInFirstWeek(int value) 设置一年中第一个星期所需的最少天数,比如。假设定义第一个星期包括一年第一个月的第一天,则使用值 1 调用此方法。
void setTime(Date date) 使用给定的 Date 设置此 Calendar 的时间。
void setTimeInMillis(long millis) 用给定的 long 值设置此 Calendar 的当前时间值。
void setTimeZone(TimeZone value) 使用给定的时区值来设置时区。
String toString() 返回此日历的字符串表示形式
Calendar的经常用法演示样例
1、计算某一月份的最大天数
Calendar time=Calendar.getInstance();
time.clear();
time.set(Calendar.YEAR,year);
time.set(Calendar.MONTH,i-1);//注意,Calendar对象默认一月为0
int day=time.getActualMaximum(Calendar.DAY_OF_MONTH);//本月份的天数
注:在使用set方法之前。必须先clear一下。否则非常多信息会继承自系统当前时间
2、Calendar和Date的转化
(1) Calendar转化为Date
Calendar cal=Calendar.getInstance();
Date date=cal.getTime();
(2) Date转化为Calendar
Date date=new Date();
Calendar cal=Calendar.getInstance();
cal.setTime(date);
3、格式化输出日期时间
Date date=new Date();
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
System.out.println(df.format(date));
4、计算一年中的第几星期
(1)计算某一天是一年中的第几星期
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
int weekno=cal.get(Calendar.WEEK_OF_YEAR);
(2)计算一年中的第几星期是几号
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.WEEK_OF_YEAR, 1);
cal.set(Calendar.DAY_OF_WEEK, Calendar.MONDAY);
System.out.println(df.format(cal.getTime()));
输出:
2006-01-02
5、add()和roll()的使用方法
(1) add()方法
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
cal.add(Calendar.DATE, -4);
Date date=cal.getTime();
System.out.println(df.format(date));
cal.add(Calendar.DATE, 4);
date=cal.getTime();
System.out.println(df.format(date));
输出:
2006-08-30
2006-09-03
(2)roll方法
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
cal.roll(Calendar.DATE, -4);
date=cal.getTime();
System.out.println(df.format(date));
cal.roll(Calendar.DATE, 4);
date=cal.getTime();
System.out.println(df.format(date));
输出:
2006-09-29
2006-09-03
可见。roll()方法在本月内循环,一般使用add()方法。
6、计算两个随意时间中间的间隔天数
(1)传进Calendar对象
/
**计算两个时间之间相隔天数
* @param startday 開始时间
* @param endday 结束时间
* @return
*/
public int getIntervalDays(Calendar startday,Calendar endday){
//确保startday在endday之前
if(startday.after(endday)){
Calendar cal=startday;
startday=endday;
endday=cal;
}
//分别得到两个时间的毫秒数
long sl=startday.getTimeInMillis();
long el=endday.getTimeInMillis();
long ei=el-sl;
//依据毫秒数计算间隔天数
return (int)(ei/(1000*60*60*24));
}
(2)传进Date对象
/**计算两个时间之间相隔天数
* @param startday 開始时间
* @param endday 结束时间
* @return
*/
public int getIntervalDays(Date startday,Date endday){
//确保startday在endday之前
if(startday.after(endday)){
Date cal=startday;
startday=endday;
endday=cal;
}
//分别得到两个时间的毫秒数
long sl=startday.getTime();
long el=endday.getTime();
long ei=el-sl;
//依据毫秒数计算间隔天数
return (int)(ei/(1000*60*60*24));
}
同理,能够用同样的方法计算出随意两个时间相隔的小时数。分钟数,秒钟数等
注:以上方法是全然按时间计算。有时并不能令人惬意,如:
startday="2006-10-11 20:00:00" endday="2006-10-12 8:00:00"
计算结果为0,可是我们或许相让计算结果变为1,此时能够用例如以下方法实现:
在传參之前,先设定endday的时间,如:
endday.set(Calendar.HOUR_OF_DAY, 23);
endday.set(Calendar.MINUTE, 59);
endday.set(Calendar.SECOND, 59);
endday.set(Calendar.MILLISECOND, 59);
这样再传进去startday,endday,则结果就如我们所愿了。只是。假设嫌以上方法麻烦,能够參考下面方法:
(3)改进精确计算相隔天数的方法
public int getDaysBetween (Calendar d1, Calendar d2) {
if (d1.after(d2)) { // swap dates so that d1 is start and d2 is end
java.util.Calendar swap = d1;
d1 = d2;
d2 = swap;
}
int days = d2.get(Calendar.DAY_OF_YEAR) - d1.get(Calendar.DAY_OF_YEAR);
int y2 = d2.get(Calendar.YEAR);
if (d1.get(Calendar.YEAR) != y2) {
d1 = (Calendar) d1.clone();
do {
days += d1.getActualMaximum(Calendar.DAY_OF_YEAR);//得到当年的实际天数
d1.add(Calendar.YEAR, 1);
} while (d1.get(Calendar.YEAR) != y2);
}
return days;
}
demo
package cn.outofmemory.codes.Date; import java.util.Calendar;
import java.util.Date; public class CalendarDemo {
public static void main(String[] args) {
Calendar calendar=Calendar.getInstance();
calendar.setTime(new Date());
System.out.println("如今时间是:"+new Date());
String year=String.valueOf(calendar.get(Calendar.YEAR));
String month=String.valueOf(calendar.get(Calendar.MONTH)+1);
String day=String.valueOf(calendar.get(Calendar.DAY_OF_MONTH));
String week=String.valueOf(calendar.get(Calendar.DAY_OF_WEEK)-1);
System.out.println("如今时间是:"+year+"年"+month+"月"+day+"日,星期"+week);
long year2009=calendar.getTimeInMillis();
calendar.set(1989,9,26);//这里与真实的月份之间相差1
long year1989=calendar.getTimeInMillis();
long days=(year2009-year1989)/(1000*60*60*24);
System.out.println("今天和1989年10月26日相隔"+days+"天,"+"也就是说我在这个漂亮的星球上已经幸福的生活了"+days+"天。"); }
}
Calendar类经常用法 日期间的转换 set方法有巨坑的更多相关文章
- day319 1、正则表达式的定义及使用 2、Date类的用法 3、Calendar类的用法
1.正则表达式的定义及使用2.Date类的用法3.Calendar类的用法 一.正则表达式 ###01正则表达式的概念和作用* A: 正则表达式的概念和作用* a: 正则表达式的概述* 正则表达式也是 ...
- Android中Calendar类的用法总结
Calendar是Android开发中需要获取时间时必不可少的一个工具类,通过这个类可以获得的时间信息还是很丰富的,下面做一个总结,以后使用的时候就不用总是去翻书或者查资料了. 在获取时间之前要先获得 ...
- Java Calendar类简单用法
我的技术博客经常被流氓网站恶意爬取转载.请移步原文:http://www.cnblogs.com/hamhog/p/3832307.html,享受整齐的排版.有效的链接.正确的代码缩进.更好的阅读体验 ...
- java成神之——date和calendar日期的用法
date和calendar日期的用法 util的data转换成sql的data 创建Date对象 格式化 Instant ChronoUnit LocalTime LocalDate LocalDat ...
- java:calendar类及一些比较实用的utils(一)
在java编程中经常会用到时间日期的计算.比较.格式化等等操作,刚开始接触Calendar类时,还是在初学习期间,小小白一枚,看着这个好复杂,懒惰心理作祟也就没有怎么去学习,后来在项目中经常用到,索性 ...
- Java Calendar类的使用总结
在实际项目当中,我们经常会涉及到对时间的处理,例如登陆网站,我们会看到网站首页显示XXX,欢迎您!今天是XXXX年....某些网站会记录下用户登陆的时间,比如银行的一些网站,对于这些经常需要处理的问题 ...
- 处理时间的类 —— System类、Date类 、SimpleDateFormat类 与 Calendar类
在我们以往的编程中,就有过通过运行前和运行后时间差来判断时间复杂度的例子,再扯得远一点,我们在C语言中制造随机数的操作,也要用到有关时间的函数.而且,在我们未来的编程中,也会时不时要用到能够读取当前时 ...
- Java Calendar类的使用总结【转】
感谢!原文地址:https://www.cnblogs.com/huangminwen/p/6041168.html Java Calendar类的使用总结 在实际项目当中,我们经常会涉及到对时间的处 ...
- 使用 Date 和 SimpleDateFormat 类表示时间、Calendar类和Math类
一. Date 和 SimpleDateFormat类表示时间 在程序开发中,经常需要处理日期和时间的相关数据,此时我们可以使用 java.util 包中的 Date 类.这个类最主要的作用就是获取当 ...
随机推荐
- [NOIP2013 花匠] 新人解题报告
本来按照老师的要求,我学OI的第一份解题报告应是在寒假完成的关于数据结构的基础题,但由于身体原因当时未能完成,那么就在省选赛前临时写几篇吧…… 题目描述 花匠栋栋种了一排花,每株花都有自己的高度.花儿 ...
- Leveldb源码解析之Bloom Filter
Bloom Filter,即布隆过滤器,是一种空间效率很高的随机数据结构. 原理:开辟m个bit位数组的空间,并全部置零,使用k个哈希函数将元素映射到数组中,相应位置1.如下图,元素K通过哈希函数h1 ...
- Java程序运行时内存划分
1.Java程序跨平台运行的原因 主要原因是:各种平台的JVM和字节码文件 Java源程序--具体平台的机器代码文件---被编译器翻译成平台无关的Class文件,又用特定JVM运行字节码文件,JVM在 ...
- [转]tx:advice标签简介
<Spring高级程序设计>第16章事务管理,通过本章的学习,你知道了如何使用Spring去管理事务,而这种方式几乎不会对你的源代码产生任何影响.你现在知道了如何使用本地和全局事务,并知道 ...
- cas协议,以及tomcat搭建cas服务器
1. CAS 简介 1.1. What is CAS ? CAS ( Central Authentication Service ) 是 Yale 大学发起的一个企业级的.开源的项目,旨 ...
- Kubernetes下的Redis主从配置架构
文章看了一大堆,但都是直接从各种地方直接拉master,slave镜像,没有交代这些镜像如何构建出来的 好把,我这篇就讲讲这些master,slave镜像如何做成. 先得找到一个标准的redis镜像, ...
- javascript 检测浏览器类型和版本的代码
方法1:对象/特征检测法 该方法是一种判断浏览器能力(而非浏览器的确切型号)的通用方法.大部分JS专家认为这个方法最合适,因为他们认为按照该方法所编写的脚本是经得起未来考验的. //获取IE浏览器的版 ...
- phpcms v9 wap手机门户站点内容页添加上一篇、下一篇的方法
PHP源码修改:打开 phpcms\modules\wap\index.php 文件找到if(!$r || $r['status'] != 99) showmessage(L('info_does_n ...
- 集合系列之fail-fast 与fail-safe 区别
一:快速失败(fail—fast) 在用迭代器遍历一个集合对象时,如果遍历过程中对集合对象的内容进行了修改(增加.删除.修改),则会抛出Concurrent Modification Exceptio ...
- Deep Learning论文笔记之(三)单层非监督学习网络分析
Deep Learning论文笔记之(三)单层非监督学习网络分析 zouxy09@qq.com http://blog.csdn.net/zouxy09 自己平时看了一些论文,但老感 ...