这期内容当中小编将会给大家带来有关如何在Java中使用Calendar类中的set()方法,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。
代码如下:
import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; public class TestCalendar { public static void main(String[] args) { String s = new SimpleDateFormat("yyyy-MM-dd") .format(getLastDay(2017, 9)); System.out.println(s); } public static Date getLastDay(int year, int month) { //获取Calendar类的实例 Calendar c = Calendar.getInstance(); //设置年份 c.set(Calendar.YEAR, year); //设置月份,因为月份从0开始,所以用month - 1 c.set(Calendar.MONTH, month - 1); //获取当前时间下,该月的最大日期的数字 int lastDay = c.getActualMaximum(Calendar.DAY_OF_MONTH); //将获取的最大日期数设置为Calendar实例的日期数 c.set(Calendar.DAY_OF_MONTH, lastDay); return c.getTime(); } }
刚开始使用这个方法的时候,很正常。后来在10月31号(这个日期很重要)当天测试的时候,传递的参数时2017年9月,即上面的代码,但是结果却出现的了问题,结果如下图:
本来该是2017-09-30,可是结果却是2017-10-01,我原先测试过,这个方法是没有问题的,可是出了这样的问题。后来我断点测试,在刚获取到Calendar实例的时候,实例中的字段值如下图:
但是发现在执行完
c.set(Calendar.MONTH, month - 1);
这行的代码的时候,Calendar的实例中,MONTH字段的值不是我预想中的8(月份字段从0开始),而是9,而且DAY_OF_MONTH字段的值从31变成了1,如下图所示:
因此,可以判断Calendar实例获取到的时候,是10月31号,实例中的DAY_OF_MONTH的值是31,当把MONTH字段的值设置为8后,因为9月份只有30天,那DAY_OF_MONTH的值就多1,会自动向后顺延1天,变成了2017-10-01 。
但是,还是有其他的问题,因为下面还执行了
c.set(Calendar.DAY_OF_MONTH, lastDay);
这句代码,最后的日期应该是2017-10-31才对,但是run的结果却是2017-10-01,debug的结果是2017-10-31 。
我第一感觉认为Calendar类是不是存在线程安全问题,可是后来一想就觉得不对,毕竟我只是在主线程中运行,没有多线程,并不存在这个问题。
第二天我又尝试了下,发现了问题的原因,如上面的最后一张图所示,在debug的过程中,我用IDEA的watches功能查看了Calendar实例的字段值,用了get()方法,如果我删除掉这几个get方法之后,发现run和debug的值是一样的,都是2017-10-01,说明问题出在get()方法上。
因此,可以做如下修改:
在代码中,直接打印变量c的值,可以发现,在调用get()方法之前,变量c的各字段值是set()方法设置的,但是并没有对其进行验证计算,在调用get()方法的过程中,会对各字段验证计算。我查看了部分源码,在调用get(),add(),getTime()等方法的过程中,底层都会调用computeTime()方法,对各字段的时间验证计算。
另外,又做了一个demo测试,以佐证上面的结论,如下:
import java.text.SimpleDateFormat; import java.util.Calendar; public class TestCalendar2 { public static void main(String[] args) { Calendar c = Calendar.getInstance(); c.set(Calendar.MONTH, 8); //将月份设置为9月 c.set(Calendar.DAY_OF_MONTH, 32); //将日期设置为32 System.out.println(c); //直接打印Calendar实例,不使用getTime()方法 c.get(Calendar.MONTH); System.out.println(c); } }
结果如下:
即使设置的DAY_OF_MONTH值是明显非法的,但是并不会在调用get()方法之前进行计算进位。
在查询问题的过程中,也看到了其他的一些问题,下面对add(),set(),roll()方法的区别做了解释:
示例代码:
Calendar c = Calendar.getInstance(); c.set(2014, Calendar.MARCH, 31); c.add(Calendar.MONTH, 13); System.out.println(c.getTime());// 2015-04-30 c.set(2014, Calendar.MARCH, 31); c.set(Calendar.MONTH, c.get(Calendar.MONTH) + 13); System.out.println(c.getTime());// 2015-05-01 c.set(2014, Calendar.MARCH, 31); c.roll(Calendar.MONTH, 13); System.out.println(c.getTime());//2014-04-30
ADD方法
以调整的单位为基点(本例中为月),较大的单位(年)会发生借位、进位。 较小的单位会往小调整。
本例中,2014-03-31,加上13个月,年份会进位为2015年。 4月31日是不存在的,所以往小调整为4月30日。
比较典型的运用场景是,日历的按月切换。
当前日期为2014-03-31,点击【下一月】按钮时,日期会变成2014-04-30.
SET方法
所有的单位都会往大调整。
本例中,2014-03-31,加上13个月,年份会进位为2015年。 4月31日是不存在的,所以往大调整为5月1日
ROLL方法
以调整的单位为基点(本例中为月),较大的单位(年)不会发生改变。 较小的单位会往小调整。
本例中,2014-03-31,加上13个月,年份依然为2014年。 4月31日是不存在的,所以往小调整为4月30日。
日会根据年、月来判断出日的取值范围,然后在1~31之间无限循环滚动,但并不会影响到年、月的值。
总结三点:
1、add() 有两条规则:
a)当被修改的字段超出它的取值范围时,那么比它大的字段会自动修正。
b)如果比它小的字段是不可变的/不在取值范围内(由 Calendar 的实现类决定),那么该小字段会修正到变化最小的值。
2、Roll() 的规则只有第二条
当被修改的字段超出它的取值范围时,那么比它大的字段不会被修正。比它小的字段会修正到变化最小的值。
3、Set()
比被修改的字段大的字段会根据字段是增大还是减小自动改变大小,比被修改字段小的字段如果是不可变的/不在取值范围内,会自动增大到变化最小的值。
回到最初的问题,获取指定年份和月份的最大的日期的方法要怎么办?
方法可以改为:
public static Date getLastDay(int year, int month) { Calendar c = Calendar.getInstance(); //获取Calendar类的实例 c.clear(); c.set(Calendar.YEAR, year); //设置年份 c.set(Calendar.MONTH, month - 1); //设置月份,因为月份从0开始,所以用month - 1 int lastDay = c.getActualMaximum(Calendar.DAY_OF_MONTH); //获取当前时间下,该月的最大日期的数字 c.set(Calendar.DAY_OF_MONTH, lastDay); //将获取的最大日期数设置为Calendar实例的日期数 return c.getTime(); //返回日期 }
上述就是小编为大家分享的如何在Java中使用Calendar类中的set()方法了,如果刚好有类似的疑惑,不妨参照上述分析进行理解。如果想知道更多相关知识,欢迎关注亿速云行业资讯频道。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。