电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

解決Java Calendar類(lèi)set()方法的陷阱

瀏覽:91日期:2022-09-05 08:30:47

在項(xiàng)目中,需要獲取指定年份和月份的最后一天。我在網(wǎng)上找到了一個(gè)用Calendar類(lèi)獲取的方法,代碼如下:

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類(lèi)的實(shí)例Calendar c = Calendar.getInstance();//設(shè)置年份c.set(Calendar.YEAR, year);//設(shè)置月份,因?yàn)樵路輳?開(kāi)始,所以用month - 1c.set(Calendar.MONTH, month - 1);//獲取當(dāng)前時(shí)間下,該月的最大日期的數(shù)字int lastDay = c.getActualMaximum(Calendar.DAY_OF_MONTH);//將獲取的最大日期數(shù)設(shè)置為Calendar實(shí)例的日期數(shù)c.set(Calendar.DAY_OF_MONTH, lastDay); return c.getTime();}}

剛開(kāi)始使用這個(gè)方法的時(shí)候,很正常。后來(lái)在10月31號(hào)(這個(gè)日期很重要)當(dāng)天測(cè)試的時(shí)候,傳遞的參數(shù)時(shí)2017年9月,即上面的代碼,但是結(jié)果卻出現(xiàn)的了問(wèn)題,結(jié)果如下圖:

解決Java Calendar類(lèi)set()方法的陷阱

本來(lái)該是2017-09-30,可是結(jié)果卻是2017-10-01,我原先測(cè)試過(guò),這個(gè)方法是沒(méi)有問(wèn)題的,可是出了這樣的問(wèn)題。后來(lái)我斷點(diǎn)測(cè)試,在剛獲取到Calendar實(shí)例的時(shí)候,實(shí)例中的字段值如下圖:

解決Java Calendar類(lèi)set()方法的陷阱

但是發(fā)現(xiàn)在執(zhí)行完

c.set(Calendar.MONTH, month - 1);

這行的代碼的時(shí)候,Calendar的實(shí)例中,MONTH字段的值不是我預(yù)想中的8(月份字段從0開(kāi)始),而是9,而且DAY_OF_MONTH字段的值從31變成了1,如下圖所示:

解決Java Calendar類(lèi)set()方法的陷阱

因此,可以判斷Calendar實(shí)例獲取到的時(shí)候,是10月31號(hào),實(shí)例中的DAY_OF_MONTH的值是31,當(dāng)把MONTH字段的值設(shè)置為8后,因?yàn)?月份只有30天,那DAY_OF_MONTH的值就多1,會(huì)自動(dòng)向后順延1天,變成了2017-10-01 。

但是,還是有其他的問(wèn)題,因?yàn)橄旅孢€執(zhí)行了

c.set(Calendar.DAY_OF_MONTH, lastDay);

這句代碼,最后的日期應(yīng)該是2017-10-31才對(duì),但是run的結(jié)果卻是2017-10-01,debug的結(jié)果是2017-10-31 。

我第一感覺(jué)認(rèn)為Calendar類(lèi)是不是存在線(xiàn)程安全問(wèn)題,可是后來(lái)一想就覺(jué)得不對(duì),畢竟我只是在主線(xiàn)程中運(yùn)行,沒(méi)有多線(xiàn)程,并不存在這個(gè)問(wèn)題。

第二天我又嘗試了下,發(fā)現(xiàn)了問(wèn)題的原因,如上面的最后一張圖所示,在debug的過(guò)程中,我用IDEA的watches功能查看了Calendar實(shí)例的字段值,用了get()方法,如果我刪除掉這幾個(gè)get方法之后,發(fā)現(xiàn)run和debug的值是一樣的,都是2017-10-01,說(shuō)明問(wèn)題出在get()方法上。

因此,可以做如下修改:

解決Java Calendar類(lèi)set()方法的陷阱

在代碼中,直接打印變量c的值,可以發(fā)現(xiàn),在調(diào)用get()方法之前,變量c的各字段值是set()方法設(shè)置的,但是并沒(méi)有對(duì)其進(jìn)行驗(yàn)證計(jì)算,在調(diào)用get()方法的過(guò)程中,會(huì)對(duì)各字段驗(yàn)證計(jì)算。我查看了部分源碼,在調(diào)用get(),add(),getTime()等方法的過(guò)程中,底層都會(huì)調(diào)用computeTime()方法,對(duì)各字段的時(shí)間驗(yàn)證計(jì)算。

另外,又做了一個(gè)demo測(cè)試,以佐證上面的結(jié)論,如下:

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); //將月份設(shè)置為9月c.set(Calendar.DAY_OF_MONTH, 32); //將日期設(shè)置為32System.out.println(c); //直接打印Calendar實(shí)例,不使用getTime()方法c.get(Calendar.MONTH);System.out.println(c);}}

結(jié)果如下:

解決Java Calendar類(lèi)set()方法的陷阱

即使設(shè)置的DAY_OF_MONTH值是明顯非法的,但是并不會(huì)在調(diào)用get()方法之前進(jìn)行計(jì)算進(jìn)位。

在查詢(xún)問(wèn)題的過(guò)程中,也看到了其他的一些問(wèn)題,下面對(duì)add(),set(),roll()方法的區(qū)別做了解釋?zhuān)?/p>

示例代碼:

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方法

以調(diào)整的單位為基點(diǎn)(本例中為月),較大的單位(年)會(huì)發(fā)生借位、進(jìn)位。 較小的單位會(huì)往小調(diào)整。 本例中,2014-03-31,加上13個(gè)月,年份會(huì)進(jìn)位為2015年。 4月31日是不存在的,所以往小調(diào)整為4月30日。 比較典型的運(yùn)用場(chǎng)景是,日歷的按月切換。 當(dāng)前日期為2014-03-31,點(diǎn)擊【下一月】按鈕時(shí),日期會(huì)變成2014-04-30.

SET方法

所有的單位都會(huì)往大調(diào)整。 本例中,2014-03-31,加上13個(gè)月,年份會(huì)進(jìn)位為2015年。 4月31日是不存在的,所以往大調(diào)整為5月1日

ROLL方法

以調(diào)整的單位為基點(diǎn)(本例中為月),較大的單位(年)不會(huì)發(fā)生改變。 較小的單位會(huì)往小調(diào)整。 本例中,2014-03-31,加上13個(gè)月,年份依然為2014年。 4月31日是不存在的,所以往小調(diào)整為4月30日。 日會(huì)根據(jù)年、月來(lái)判斷出日的取值范圍,然后在1~31之間無(wú)限循環(huán)滾動(dòng),但并不會(huì)影響到年、月的值。

總結(jié)三點(diǎn):

1、add() 有兩條規(guī)則: a)當(dāng)被修改的字段超出它的取值范圍時(shí),那么比它大的字段會(huì)自動(dòng)修正。 b)如果比它小的字段是不可變的/不在取值范圍內(nèi)(由 Calendar 的實(shí)現(xiàn)類(lèi)決定),那么該小字段會(huì)修正到變化最小的值。 2、Roll() 的規(guī)則只有第二條 當(dāng)被修改的字段超出它的取值范圍時(shí),那么比它大的字段不會(huì)被修正。比它小的字段會(huì)修正到變化最小的值。 3、Set() 比被修改的字段大的字段會(huì)根據(jù)字段是增大還是減小自動(dòng)改變大小,比被修改字段小的字段如果是不可變的/不在取值范圍內(nèi),會(huì)自動(dòng)增大到變化最小的值。

回到最初的問(wèn)題,獲取指定年份和月份的最大的日期的方法要怎么辦?

方法可以改為:

public static Date getLastDay(int year, int month) {Calendar c = Calendar.getInstance(); //獲取Calendar類(lèi)的實(shí)例c.clear();c.set(Calendar.YEAR, year); //設(shè)置年份c.set(Calendar.MONTH, month - 1); //設(shè)置月份,因?yàn)樵路輳?開(kāi)始,所以用month - 1int lastDay = c.getActualMaximum(Calendar.DAY_OF_MONTH); //獲取當(dāng)前時(shí)間下,該月的最大日期的數(shù)字c.set(Calendar.DAY_OF_MONTH, lastDay); //將獲取的最大日期數(shù)設(shè)置為Calendar實(shí)例的日期數(shù)return c.getTime(); //返回日期}

用clear()方法,將Calendar實(shí)例的字段和時(shí)間都設(shè)置為未定義,這樣可以解決這個(gè)問(wèn)題。

當(dāng)然網(wǎng)上也有將月份設(shè)置為下個(gè)月,然后用add(Calendar.DAY_OF_MONTH, -1)這樣的方法也可以得到結(jié)果,不過(guò)這里就不詳細(xì)介紹了。

到此這篇關(guān)于解決Java Calendar類(lèi)set()方法的陷阱的文章就介紹到這了,更多相關(guān)Java Calendar set()內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 气力输送设备_料封泵_仓泵_散装机_气化板_压力释放阀-河南锐驰机械设备有限公司 | 影像测量仪_三坐标测量机_一键式二次元_全自动影像测量仪-广东妙机精密科技股份有限公司 | 拉伸膜,PE缠绕膜,打包带,封箱胶带,包装膜厂家-东莞宏展包装 | 泥沙分离_泥沙分离设备_泥砂分离机_洛阳隆中重工机械有限公司 | 电动葫芦-河北悍象起重机械有限公司| 临时厕所租赁_玻璃钢厕所租赁_蹲式|坐式厕所出租-北京慧海通 | 自动气象站_农业气象站_超声波气象站_防爆气象站-山东万象环境科技有限公司 | 杭州翻译公司_驾照翻译_专业人工翻译-杭州以琳翻译有限公司官网 组织研磨机-高通量组织研磨仪-实验室多样品组织研磨机-东方天净 | 柴油机_柴油发电机_厂家_品牌-江苏卡得城仕发动机有限公司 | 二手光谱仪维修-德国OBLF光谱仪|进口斯派克光谱仪-热电ARL光谱仪-意大利GNR光谱仪-永晖检测 | 对夹式止回阀_对夹式蝶形止回阀_对夹式软密封止回阀_超薄型止回阀_不锈钢底阀-温州上炬阀门科技有限公司 | 5nd音乐网|最新流行歌曲|MP3歌曲免费下载|好听的歌|音乐下载 免费听mp3音乐 | 执业药师报名时间,报考条件,考试时间-首页入口 | 多米诺-多米诺世界纪录团队-多米诺世界-多米诺团队培训-多米诺公关活动-多米诺创意广告-多米诺大型表演-多米诺专业赛事 | 科箭WMS仓库管理软件-TMS物流管理系统-科箭SaaS云服务 | 陕西安玻璃自动感应门-自动重叠门-磁悬浮平开门厂家【捷申达门业】 | 安徽免检低氮锅炉_合肥燃油锅炉_安徽蒸汽发生器_合肥燃气锅炉-合肥扬诺锅炉有限公司 | 雷达液位计_超声波风速风向仪_雨量传感器_辐射传感器-山东风途物联网 | 山东PE给水管厂家,山东双壁波纹管,山东钢带增强波纹管,山东PE穿线管,山东PE农田灌溉管,山东MPP电力保护套管-山东德诺塑业有限公司 | 工控机,嵌入式主板,工业主板,arm主板,图像采集卡,poe网卡,朗锐智科 | 耐压仪-高压耐压仪|徐吉电气| 贝朗斯动力商城(BRCPOWER.COM) - 买叉车蓄电池上贝朗斯商城,价格更超值,品质有保障! | 杭州货架订做_组合货架公司_货位式货架_贯通式_重型仓储_工厂货架_货架销售厂家_杭州永诚货架有限公司 | 北京征地律师,征地拆迁律师,专业拆迁律师,北京拆迁律师,征地纠纷律师,征地诉讼律师,征地拆迁补偿,拆迁律师 - 北京凯诺律师事务所 | 河南膏药贴牌-膏药代加工-膏药oem厂家-洛阳今世康医药科技有限公司 | 威海防火彩钢板,威海岩棉复合板,威海彩钢瓦-文登区九龙岩棉复合板厂 | 超声骨密度仪-动脉硬化检测仪器-人体成分分析仪厂家/品牌/价格_南京科力悦 | 预制直埋蒸汽保温管-直埋管道-聚氨酯发泡保温管厂家 - 唐山市吉祥保温工贸有限公司 | 飞飞影视_热门电影在线观看_影视大全 | 高尔夫球杆_高尔夫果岭_高尔夫用品-深圳市新高品体育用品有限公司 | 包塑丝_高铁绑丝_地暖绑丝_涂塑丝_塑料皮铁丝_河北创筹金属丝网制品有限公司 | 茅茅虫AI论文写作助手-免费AIGC论文查重_写毕业论文降重 | 长沙一级消防工程公司_智能化弱电_机电安装_亮化工程专业施工承包_湖南公共安全工程有限公司 | 厚壁钢管-厚壁无缝钢管-小口径厚壁钢管-大口径厚壁钢管 - 聊城宽达钢管有限公司 | 工业冷却塔维修厂家_方形不锈钢工业凉水塔维修改造方案-广东康明节能空调有限公司 | 手持式3d激光扫描仪-便携式三维立体扫描仪-北京福禄克斯 | YAGEO国巨电容|贴片电阻|电容价格|三星代理商-深圳市巨优电子有限公司 | 浙江上沪阀门有限公司 | 特种电缆厂家-硅橡胶耐高温电缆-耐低温补偿导线-安徽万邦特种电缆有限公司 | 安全阀_弹簧式安全阀_美标安全阀_工业冷冻安全阀厂家-中国·阿司米阀门有限公司 | 单电机制砂机,BHS制砂机,制沙机设备,制砂机价格-正升制砂机厂家 单级/双级旋片式真空泵厂家,2xz旋片真空泵-浙江台州求精真空泵有限公司 |