亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

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

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

在項目中,需要獲取指定年份和月份的最后一天。我在網上找到了一個用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月,即上面的代碼,但是結果卻出現的了問題,結果如下圖:

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

本來該是2017-09-30,可是結果卻是2017-10-01,我原先測試過,這個方法是沒有問題的,可是出了這樣的問題。后來我斷點測試,在剛獲取到Calendar實例的時候,實例中的字段值如下圖:

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

但是發現在執行完

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

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

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

因此,可以判斷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()方法上。

因此,可以做如下修改:

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

在代碼中,直接打印變量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);}}

結果如下:

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

即使設置的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 - 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)這樣的方法也可以得到結果,不過這里就不詳細介紹了。

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

標簽: Java
相關文章:
主站蜘蛛池模板: 国产精品入口在线看麻豆 | 911国产视频 | 日本xxxx色视频在线观看 | 一级特黄a 大片免费 | 亚洲情a成黄在线观看 | 美女毛片在线看 | 国产夫妇精品自在线 | 国产亚洲精品免费 | 香蕉成人999视频 | 国产一级高清视频免费看 | 亚洲国产激情在线一区 | 美女黄色免费在线观看 | 黄色片网站免费观看 | 国产免费啪视频观看网站 | 久久成人影视 | 免费jizz在线播放视频 | 日韩午夜伦y4480私人影院 | 岛国激情片 | 黄色的毛片 | 九色精品在线 | 久久99国产亚洲精品 | 色视频一区二区三区 | 青青操网 | 免费国产阿v视频在线观看 免费国产高清精品一区在线 | 久青草国产手机视频免费观看 | 国产欧美精品一区二区三区四区 | 在线视频观看一区 | 一级做a爱片久久毛片 | 亚洲成在人线影视天堂网 | 性插影院| 成人国产精品一级毛片视频 | 精品一区二区三区中文 | 亚洲免费精品视频 | 国产一级片在线播放 | 国产一级爱c片免费观看 | 看一级黄色毛片 | 日韩黄色大片 | 亚洲制服一区 | 自拍视频在线观看视频精品 | 精品国产91在线网 | 欧美在线一区二区三区 |