java中l(wèi)ist使用時(shí)需避免的場景總結(jié)
眾所周知,Java
為開發(fā)者提供了多種集合類的實(shí)現(xiàn)。而Java
的集合類通常被分為兩大類:Map
和Collection
。進(jìn)一步,Collection
又分為三個(gè)子類,包括List、Set和Queue
。其中,可以幾乎所有業(yè)務(wù)代碼都需要用到List
,因此其也被認(rèn)為是最為是集合中最重要的一個(gè)結(jié)構(gòu)。
但List
的錯(cuò)誤使用也會導(dǎo)致諸多問題,今天我們就來看一看幾個(gè)錯(cuò)誤使用List
的場景。
前言
在日常的Java
面試中,ArrayList
絕對可以算得上是Java
后端面試中的一個(gè)???。為此網(wǎng)上也有很多文章來對ArrayList
的源碼進(jìn)行分析,其實(shí)重點(diǎn)歸結(jié)下來無非以下幾點(diǎn)
ArrayList
中元素增、刪操作的原理;ArrayList
中的擴(kuò)容機(jī)制,元素拷貝原理;ArrayList
線程安全性以及fast-fail
機(jī)制等。
在面試中可能你對于上述問題可以做到對答如流,但是真正落實(shí)到實(shí)際開發(fā)中你能保證寫出沒有Bug
的代碼嗎?對于這個(gè)問題不同的開發(fā)者
可能有著不同的答案,但無論你的答案是什么,筆者還是希望你能花幾分鐘讀一讀后續(xù)的內(nèi)容,相信讀完你一定會有所收獲~~~
刪除元素時(shí)傳遞的參數(shù)到底是什么
在開始分析之前,我們先來看一段代碼:
@Test public void testStrList() { ArrayList<String> strs = new ArrayList<>(); strs.add("hello"); strs.add("coding"); strs.add("word"); strs.remove(1); log.info("Arrays after is [{}] ", Arrays.toString(strs.toArray())); }
上述代碼邏輯很簡單,無非就是向strs
中添加三個(gè)字符串
信息,然后調(diào)用remove
方法進(jìn)行刪除,那此時(shí)上述代碼會打印出什么內(nèi)容呢?此時(shí),相信你肯定不假思索的回答出肯定是Arrays after is [[hello, word]]
。但是當(dāng)我拿出如下這段代碼,閣下又該如何應(yīng)對呢?
@Test public void testList() { ArrayList<Integer> nums = new ArrayList<>(); for (int i = 0 ; i < 5 ; i ++) { nums.add(5-i); } // <1> nums.remove(new Integer(3)); nums.remove(3); log.info("Arrays after is [{}] ", Arrays.toString(nums.toArray())); }
對于上述代碼,筆者有兩個(gè)疑問:
- 上述代碼會輸出什么呢?
- 如果將上述代碼
nums.remove(3)
注釋掉,而將<1>
處代碼注釋打開,其輸出結(jié)果又是什么呢?
這個(gè)問題歸根到底本質(zhì)就是對于ArrayList
中remove
方法的理解,可能面試
時(shí)你對于ArrayList
的remove
方法也能侃侃而談,但面對如此場景你還能不假思索的回答出來嗎?
有疑惑也別慌,接下來我們就來看看List
中關(guān)于remove
的方法有什么樣的定義:
List # remove
public interface List<E> extends Collection<E> { ...... boolean remove(Object o); ...... E remove(int index); ...... }
可以看到,在List
接口中,對于remove
方法進(jìn)行了重載,其支持兩種類型的傳參:一種是傳入基本類型
;另一種則需要傳入一個(gè)對象
。進(jìn)一步,ArrayList
中對于不同參數(shù)下remove
方法的執(zhí)行邏輯也是有差異的。具體來看:
- 當(dāng)
remove
中傳入的參數(shù)為int
時(shí),則會刪除參數(shù)
對應(yīng)索引
中的元素; - 當(dāng)
remove
中傳入的參數(shù)為Object
時(shí),則會遍歷列表中的元素
,進(jìn)而刪除列表中的對應(yīng)元素
。
至此,上述代碼執(zhí)行后的結(jié)果其實(shí)已經(jīng)很明顯了。當(dāng)執(zhí)行nums.remove(3)
時(shí),其會刪除對應(yīng)索引下的的元素,即刪除列表中的元素2
;而當(dāng)執(zhí)行nums.remove(new Integer(3))
時(shí),則會刪除列表中的元素3
。
可能你會覺得,這里無非就是remove
方法有一個(gè)重載邏輯罷了,有何難?你無非就是比我看的時(shí)候細(xì)心
了一點(diǎn)罷了,這又能算的上什么呢?
如果你有這樣的想法,那不妨來看看如下這段代碼,想一想下列代碼最終輸出的size
是多少呢?
public void testStrList() { ArrayList<String> strs = new ArrayList<>(); strs.add("hello"); strs.add("coding"); strs.add("word"); // 定義匹配規(guī)則 ArrayList<String> mapStrs = new ArrayList<>(Arrays.asList("hello")); // 存儲適配信息坐標(biāo) ArrayList<Integer> index = new ArrayList<>(); for (int i = 0 ; i < strs.size() ; i ++) { if (mapStrs.contains(strs.get(i))) { index.add(i); } } // 刪除匹配信息 for (int i = 0 ; i < index.size() ; i ++) { strs.remove(index.get(i)); } log.info("strs removed size is [{}] ", strs.size()); }
上述代碼最終會輸出strs removed size is 3
,不知你的答案是否是這樣的?至于為何是這樣,筆者在此就不進(jìn)行分析了。這算是筆者留的一個(gè)思考題,歡迎你在評論區(qū)留言。
Arrays.asList你用對了嗎
眾所周知,Arrays.asList
是 Java
中的一個(gè)靜態(tài)方法,它的主要作用是將一個(gè)數(shù)組轉(zhuǎn)換成一個(gè)固定大小的 List
。 那下述兩種方式構(gòu)建的List
有什么差異呢?
int[] arr = {1, 2, 3}; // <1> 傳入數(shù)組構(gòu)建List List listArray = Arrays.asList(arr); // <2> 通過參數(shù)構(gòu)建List List list = Arrays.asList(1,2,3);
執(zhí)行代碼你會發(fā)現(xiàn),方法<1>
這樣初始化的 List
并不是我們期望的包含 3
個(gè)數(shù)字的 List
。而其會創(chuàng)建一個(gè)元素類型為[]int
的List
。換言之,其最終會生成的 List
是一個(gè)元素個(gè)數(shù)為 1
的整型數(shù)組。
那為什么會這樣?我們來看下其源碼:
public static List asList(T... a) { return new ArrayList<>(a); }
進(jìn)一步,其在構(gòu)建ArrayList
時(shí)的邏輯如下所示:
通過追蹤源碼可以看到,Arrays.asList
方法傳入的是一個(gè)泛型T
類型可變參數(shù),而我們傳入的int[]
最終作為了一個(gè)對象成為了泛型類型 T
。當(dāng)然,這個(gè)問題更深層的原因在于:Jdk
在對象的封箱/拆箱只支持將 int
裝箱為 Integer
,卻不能把 int 數(shù)組
裝箱為 Integer 數(shù)組
。進(jìn)一步,如下代碼則不會存在此問題。
Integer[] arr2 = {1, 2, 3}; List list = Arrays.asList(arr2);
總之, Arrays.asList
不能直接使用來轉(zhuǎn)換基本類型數(shù)組。 進(jìn)一步,既然Arrays.asList
可以得到一個(gè)List
,那對其進(jìn)行add、remove
等操作是不是沒啥問題?。例如如下這樣
Integer[] arr2 = {1, 2, 3}; List list = Arrays.asList(arr2); list.add(3,4)
如果運(yùn)行上述代碼你會喜提一個(gè)UnsupportedOperationException
。換言之,為list
中的3
號位置新增整型 4
的操作失敗了。造成這一問題的在于:Arrays.asList
返回的List
不支持增刪操作。即Arrays.asList
返回的 List
并不是 我們期望的 java.util.ArrayList
。其最終會返回 Arrays
的內(nèi)部類 ArrayList
。而該類并沒有覆寫父類AbstractList
的add
方法,而父類中add
方法的實(shí)現(xiàn),就是拋出 UnsupportedOperationException
。相關(guān)代碼如下:
AbstractList # add
public void add(int index, E element) { throw new UnsupportedOperationException(); }
總結(jié)
至此,我們拋磚引玉的介紹了使用List
時(shí)可能遇到的一些小問題。事實(shí)上,除了上述討論的問題外List
在使用切片功能也可能導(dǎo)致一些意想不到的OOM
異常,在此筆者就不逐一進(jìn)行分析了,感興趣的讀者可翻閱相關(guān)資料進(jìn)行了解。
或許,List
使用過程中的這些小細(xì)節(jié)在你看來無關(guān)緊要,但細(xì)節(jié)決定成敗。當(dāng)然,這也是一個(gè)仁者見仁智者的問題,筆者在此就不贅述了。
到此這篇關(guān)于java中l(wèi)ist使用時(shí)需避免的場景總結(jié)的文章就介紹到這了,更多相關(guān)java list內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用@Validated注解進(jìn)行參數(shù)驗(yàn)證的方法
這篇文章主要介紹了Java使用@Validated注解進(jìn)行參數(shù)驗(yàn)證的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08Java數(shù)據(jù)結(jié)構(gòu)之?dāng)?shù)組(動力節(jié)點(diǎn)之Java學(xué)院整理)
這篇文章主要介紹了Java數(shù)據(jù)結(jié)構(gòu)之?dāng)?shù)組(動力節(jié)點(diǎn)之Java學(xué)院整理)的相關(guān)資料,包括創(chuàng)建和內(nèi)存分配,數(shù)組封裝后的使用等,需要的朋友參考下吧2017-04-04Java反射機(jī)制,反射相關(guān)API,反射API使用方式(反射獲取實(shí)體類字段名和注解值)
這篇文章主要介紹了Java反射機(jī)制,反射相關(guān)API,反射API使用方式(反射獲取實(shí)體類字段名和注解值),具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07SpringCloud OpenFeign與Ribbon客戶端配置詳解
在springcloud中,openfeign是取代了feign作為負(fù)載均衡組件的,feign最早是netflix提供的,他是一個(gè)輕量級的支持RESTful的http服務(wù)調(diào)用框架,內(nèi)置了ribbon,而ribbon可以提供負(fù)載均衡機(jī)制,因此feign可以作為一個(gè)負(fù)載均衡的遠(yuǎn)程服務(wù)調(diào)用框架使用2022-11-11Spring+SpringMVC+MyBatis深入學(xué)習(xí)及搭建(三)之MyBatis全局配置文件解析
這篇文章主要介紹了Spring+SpringMVC+MyBatis深入學(xué)習(xí)及搭建(三)之MyBatis全局配置文件解析,需要的朋友可以參考下2017-05-05淺析fastjson2時(shí)間序列化和反序列化的簡單使用
在項(xiàng)目中將fastjson升級為fastjson2后,我們遇到了一些與fastjson不完全兼容的問題,所以本文就來探討下fastjson2的時(shí)間序列化和反序列化的簡單使用吧2025-01-01