解决Java Calendar类set()方法的陷阱

在项目中,需要获取指定年份和月份的最后一天。我在网上找到了一个用Calendar类获取的方法,代码如下:

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();           //返回日期
}

用clear()方法,将Calendar实例的字段和时间都设置为未定义,这样可以解决这个问题。

当然网上也有将月份设置为下个月,然后用add(Calendar.DAY_OF_MONTH, -1)这样的方法也可以得到结果,不过这里就不详细介绍了。

到此这篇关于解决Java Calendar类set()方法的陷阱的文章就介绍到这了,更多相关Java Calendar set()内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • JavaWeb项目FullCalendar日历插件使用的示例代码

    本文介绍了JavaWeb项目FullCalendar日历插件使用的示例代码,分享给大家,具体如下: 使用FullCalendar需要引用的文件 1.css文件 2.js文件 <link href="${base}/assets/global/plugins/fullcalendar/fullcalendar.min.css" rel="external nofollow" rel="stylesheet" type="text/c

  • Java中Date,Calendar,Timestamp的区别以及相互转换与使用

    1 Java.util.Date包含年.月.日.时.分.秒信息. 复制代码 代码如下: // String转换为DateString dateStr="2013-8-13 23:23:23";String pattern="yyyy-MM-dd HH:mm:ss";DateFormate dateFormat=new SimpleDateFormat(pattern);Date date=dateFormat.parse(dateStr);date=dateForm

  • Java时间类Date类和Calendar类的使用详解

    起因:写代码的时候经常会用到获取当前时间戳和日期,现总结如下 public void testDate() { //SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd");//设置日期格式 Date date = new Date(); String dateString = date.toString(); long times = date.getTime(); System.out.println("date.t

  • Java中Calendar时间操作常用方法详解

    本文实例为大家分享了Calendar时间操作常用方法,具体内容如下 package test; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; /** * Date和Calendar常用方法,Date很多方法已经弃用,因此以Calendar为主 * * @author tuzongxun123 * */ public class DateAndCalendarTest { p

  • Java中的日期和时间类以及Calendar类用法详解

    Java日期和时间类简介 Java 的日期和时间类位于 java.util 包中.利用日期时间类提供的方法,可以获取当前的日期和时间,创建日期和时间参数,计算和比较时间. Date 类 Date 类是 Java 中的日期时间类,其构造方法比较多,下面是常用的两个: Date():使用当前的日期和时间初始化一个对象. Date(long millisec):从1970年01月01日00时(格林威治时间)开始以毫秒计算时间,计算 millisec 毫秒.如果运行 Java 程序的本地时区是北京时区(

  • Java Calendar类的详解及使用实例

    Java Calendar类的使用总结 在实际项目当中,我们经常会涉及到对时间的处理,例如登陆网站,我们会看到网站首页显示XXX,欢迎您!今天是XXXX年....某些网站会记录下用户登陆的时间,比如银行的一些网站,对于这些经常需要处理的问题,Java中提供了Calendar这个专门用于对日期进行操作的类,那么这个类有什么特殊的地方呢,首先我们来看Calendar的声明 public abstract class Calendar extends Objectimplements Serializ

  • Java中的Calendar日历API用法完全解析

    第一部分 Calendar介绍 Calendar 定义: public abstract class Calendar implements Serializable, Cloneable, Comparable<Calendar> {} Calendar 可以看作是一个抽象类. 它的实现,采用了设计模式中的工厂方法.表现在:当我们获取Calendar实例时,Calendar会根据传入的参数来返回相应的Calendar对象.获取Calendar实例,有以下两种方式: (1) 当我们通过 Cal

  • Java Calendar类的时间操作

    Java Calendar 类时间操作,这也许是创建日历和管理最简单的一个方案,示范代码很简单,演示了获取时间,日期时间的累加和累减,以及比较. 注意事项: Calendar 的 month 从 0 开始,也就是全年 12 个月由 0 ~ 11 进行表示. 而 Calendar.DAY_OF_WEEK 定义和值如下: Calendar.SUNDAY = 1 Calendar.MONDAY = 2 Calendar.TUESDAY = 3 Calendar.WEDNESDAY = 4 Calend

  • java时间 java.util.Calendar深入分析

    java.util.Calendar 在Java中时间的类有几个,但是随着Date被渐渐禁用,其中的方法慢慢打上了叉号,剩下能用的函数在Calendar中都已实现,而Calendar的子类GregorianCalendar又过于深入特殊日历的研究,平时我们并不会用到这个子类.我们可以相信Calendar类会是以后的主流时间类,下面我们就一起看一下Calendar类的详细内容,如果有错误欢迎大家指正. (一) 实例化 Calendar类是一个抽象类,是不能实例化的,那么这个类得到一个日历实例的方法

  • java中Calendar类用法实例详解

    本文实例讲述了java中Calendar类用法.分享给大家供大家参考,具体如下: java中的Calendar在开发中经常被忽略,这篇博客总结一下这个类,对后面项目中使用时期的时候有帮助. Calendar常量(field)的作用 Calendar cal = Calendar.getInstance(); cal.get(Calendar.DATE);//-----------------------当天 1-31 cal.get(Calendar.DAY_OF_MONTH);//------

  • Java中Date和Calendar常用方法

    在java中用到的最多的时间类莫过于 java.util.Date了, 由于Date类中将getYear(),getMonth()等获取年.月.日的方法都废弃了,所以要借助于Calendar来获取年.月.日.周等比较常用的日期格式 注意:以下代码均已在jdk1.6中测试通过,其他版本可能使用不同,请注意! Date与String的互转用法 /** * Date与String的互转用法,这里需要用到SimpleDateFormat */ Date currentTime = new Date();

随机推荐