Calendar类set()方法的“陷阱”

article/2025/9/21 14:06:54

2023年6月5日更新:我看还有开发同志在用Calendar,我想目前大部分项目jdk一般使用的都是JDK8及以上了,而且JDK11和17市场份额好像在爆发,其实更建议使用 JDK8提供的日期时间API(LocalDate和LocalDateTime),确实要更好用一些。

-------------------- 以下为原文 --------------------

在项目中,需要获取指定年份和月份的最后一天。我在网上找到了一个用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 - 1c.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);   //将日期设置为32System.out.println(c);              //直接打印Calendar实例,不使用getTime()方法c.get(Calendar.MONTH);System.out.println(c);}
}


结果如下:

即使设置的DAY_OF_MONTH值是明显非法的,但是并不会在调用get()方法之前进行计算进位。

在查询问题的过程中,也看到了其他的一些问题,这篇文章对add(),set(),roll()方法的区别做了解释:

新浪博客

回到最初的问题,获取指定年份和月份的最大的日期的方法要怎么办?

方法可以改为:

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 - 1int 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)这样的方法也可以得到结果,不过这里就不详细介绍了。


http://chatgpt.dhexx.cn/article/G7Jn1033.shtml

相关文章

日历Calendar类

Candendar类是一个抽象类,提供了一些获取当前时间,或者指定的时间的字段和一些方法,我们可以通过一些方法与字段对他进行获取当前天或者当月的一些信息。 创建一个Candendar对象 我们都知道创建一个类的对象最简单的方法是从他的构造方法入…

Java—java.util.calendar类详解

目录 一、概述 二、静态常量 三、静态方法 四、实例 五、GregorianCalendar类 一、概述 java.util.Calendar类是一个抽象类,是java日期处理的核心类之一。Calendar类为操作日历字段,及其与特定瞬间之间的转换提供了方法。日历字段包含YEAR、MONTH、…

Java学习笔记——Calendar类

Java中的日期类主要包括:Date类和Calendar类,本文中将对Calendar类进行讲解。 一、Calendar类的简介 Calendar类是一个抽象类,在实际使用时实现特定的子类的对象,只需要使用getInstance方法创建即可。 二、Calendar类中的常用字…

Calendar类

目录 一、Calendar 类概述二、Calendar类的构造方法三、Caleadar类的基本使用四、Calendar的常用方法1.public abstract void add(int filed,int amount)2.public final void set(int year,int month,int date) 一、Calendar 类概述 Calendar常常被称为日历类 可以看出Calen…

(Java Web) 提交表单 实例讲解

传输过程 表单 to Servlet 开门见山&#xff0c;一张图解释 首先设置表单的<form action"myURL/toText" method"post"> 此处的action对应web.xml中的url-pattern web.xml中需要注册好对应的Servlet method可设置post 或者 get不设置的话默认是get 后…

微信小程序提交表单

先看效果 身份选择使用了picker 具体代码&#xff1a; form.wxml <view class"modify-form"> <form bindsubmitgetForm><view class"label-list"><view class"label">姓名</view><view class"input-b…

Html提交表单的制作

Html提交表单制作 笔记整理 1. 代码 <!-- 知识总结&#xff1a;提交表单中form必须有action属性&#xff0c;表示提交地址所有提交的数据&#xff0c;input必须具有name属性&#xff0c;才能把数据提交到指定地址input按钮的文字&#xff0c;使用value表示属性input必须放…

异步提交表单

异步提交表单 异步提交表单的步骤 所谓异步提交表单&#xff0c;就是不再使用表单的提交按钮实现表单的提交功能&#xff0c;而是通过Ajax异步交互方式实现表单提交。具体实现步骤如下: 获取表单及所有表单组件对应的数据值。将所有表单组件对应的数据值拼成特定格式的字符串…

按钮提交表单

2.3 提交表单数据 ASP.NET Framework包含三个用于向服务器端提交表单的控件&#xff1a;Button、LinkButton和ImageButton。这三个控件拥有同样的功能&#xff0c;但每种控件的外观截然不同。 本节学习如何在页面中使用这三种控件。然后&#xff0c;学习如何关联客户端脚本和服…

from 表单提交

因为是转载文章 在此标明出处&#xff0c;以前有文章是转的没标明的请谅解&#xff0c;因为有些已经无法找到出处&#xff0c;或者与其它原因。 如有冒犯请联系本人&#xff0c;或删除&#xff0c;或标明出处。 因为好的文章&#xff0c;以前只想收藏&#xff0c;但连接有时候会…

HTML_表单与提交

<!-- form 标签 action 提交地址 method 提交方式get 高效 但数据会在url中显示 且传输量小post 效率较低 但数据不在url中显示 且传输量大 --> <!--name属性作为提交数据时数据的变量名--> <!--value属性作为填充值(多种意义上)--> <!--单选框/多选框中…

form表单的提交

开发工具与关键技术&#xff1a;MVC JQuery 的 form表单的提交 一、Form表单有两个属性分别是&#xff1a;“action”和“method”: Action: 的值是URL 就是当提交表单时向某个地方&#xff08;要提交到某处的地址&#xff09;发送表单数据 Method: 的值是 get和 post 就是用来…

表单提交的方法

form表单有两种属性action与method。 action属性有一个值URL。它规定当提交表单时向何处放送表单数据&#xff0c;URL有两种值&#xff1a;一种绝对URL&#xff0c;一种相对URL。 绝对URL指向其他站点(比如 srcwww.baidu.com网址)。 相对URL指向站点内的文件(比如 src&#…

form表单提交的几种方式

表单提交方式一&#xff1a;直接利用form表单提交 html页面代码&#xff1a; <!DOCTYPE html> <html> <head> <meta charset"UTF-8" /> <title>Insert title here</title> </head> <body> <form action"ht…

extremecomponents -- 文档下载依赖使用

extremecomponents – 文档下载依赖使用 jar包下载链接: https://mvnrepository.com/artifact/org.extremecomponents/extremecomponents. https://blog.csdn.net/yu__yfchun125/article/details/7655593

extremecomponents相关大全

安装要求1、Servlet 2.3 或更高2、 JDK 1.3.1 或更高 最小的Jars需求1、commons-beanutils 1.62、commons-collections 3.03、 commons-lang 2.04、 commons-logging 1.0.45、 standard 1.0.2 PDF 导出要用到的包:1、 avalon-framework 4.02、batik 1.5-fop-0.20-53、 fop 0.2…

ExtremeComponents源码解析(一)

一、前言 因参与公司框架改造&#xff0c;在负责前端table组件选型时&#xff0c;原本选了jqGrid和Bootstraptable作为备选方案&#xff0c;评审会上&#xff0c;武哥提了EXtremeComponents&#xff0c;让我也去了解下&#xff0c;看下合不合适&#xff0c;在此机缘下&#xff…

eXtremeComponents的eXtremeTable分页特性

<script type"text/javascript"> </script> <script type"text/javascript" src"http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script> eXtremeComponents的eXtremeTable是一套很好的分页标签&#xf…

Android Dialog

创建对话框 Showing a Dialog 显示对话框 Dismissing a Dialog 解除对话框 Using dismiss listeners 使用解除监听器Creating an AlertDialog 创建警告对话框 Adding buttons 增加按钮 Adding a list 增加列表 Adding checkboxes and radio buttons 增加单选框和复选框Creating…

关于DialogResult

在程序中&#xff0c;经常会弹出一个对话框来让用户填写一些信息&#xff0c;填写完成后&#xff0c;当用户点击“确定”按钮后&#xff0c;在主窗体中进行其他的处理。比如一个简单的例子&#xff0c;在主窗体中有一个菜单&#xff0c;是“增加用户”&#xff0c;当点击这个菜…