您好,登錄后才能下訂單哦!
這篇文章主要講解了“Java開(kāi)發(fā)人員有哪些最常犯的錯(cuò)誤”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Java開(kāi)發(fā)人員有哪些最常犯的錯(cuò)誤”吧!
Array轉(zhuǎn)ArrayList
當(dāng)需要把Array轉(zhuǎn)成ArrayList的時(shí)候,開(kāi)發(fā)人員經(jīng)常這樣做:
List<String> list = Arrays.asList(arr);
Arrays.asList()會(huì)返回一個(gè)ArrayList,但是要特別注意,這個(gè)ArrayList是Arrays類(lèi)的靜態(tài)內(nèi)部類(lèi),并不是java.util.ArrayList類(lèi)。java.util.Arrays.ArrayList類(lèi)實(shí)現(xiàn)了set(), get(),contains()方法,但是并沒(méi)有實(shí)現(xiàn)增加元素的方法(事實(shí)上是可以調(diào)用add方法,但是沒(méi)有具體實(shí)現(xiàn),僅僅拋出UnsupportedOperationException異常),因此它的大小也是固定不變的。為了創(chuàng)建一個(gè)真正的java.util.ArrayList,你應(yīng)該這樣做:
ArrayList<String> arrayList = new ArrayList<String>(Arrays.asList(arr));
ArrayList的構(gòu)造方法可以接收一個(gè)Collection類(lèi)型,而java.util.Arrays.ArrayList已經(jīng)實(shí)現(xiàn)了該接口。
判斷一個(gè)數(shù)組是否包含某個(gè)值
開(kāi)發(fā)人員經(jīng)常這樣做:
Set<String> set = new HashSet<String>(Arrays.asList(arr));return set.contains(targetValue);
以上代碼可以正常工作,但是沒(méi)有必要將其轉(zhuǎn)換成set集合,將一個(gè)List轉(zhuǎn)成Set需要額外的時(shí)間,其實(shí)我們可以簡(jiǎn)單的使用如下方法即可:
Arrays.asList(arr).contains(targetValue);
或者
for(String s: arr){ if(s.equals(targetValue)) return true;}return false;
第一種方法可讀性更強(qiáng)。
在循環(huán)內(nèi)部刪除List中的一個(gè)元素
考慮如下代碼,在迭代期間刪除元素:
ArrayList<String> list = new ArrayList<String>(Arrays.asList("a", "b", "c","d"));for (int i = 0; i < list.size(); i++) { list.remove(i);}System.out.println(list);
結(jié)果打?。?/p>
[b, d]
在上面這個(gè)方法中有一系列的問(wèn)題,當(dāng)一個(gè)元素被刪除的時(shí)候,list大小減小,然后原先索引指向了其它元素。所以如果你想在循環(huán)里通過(guò)索引來(lái)刪除多個(gè)元素,將不會(huì)正確工作。
你也許知道使用迭代器是在循環(huán)里刪除元素的正確方式,或許你也知道foreach循環(huán)跟迭代器很類(lèi)似,但事實(shí)情況卻不是這樣,如下代碼:
ArrayList<String> list = new ArrayList<String>(Arrays.asList("a", "b", "c","d"));for (String s : list) { if (s.equals("a")) list.remove(s);}
將拋出ConcurrentModificationException異常。
然而接下來(lái)的代碼卻是OK的:
ArrayList<String> list = new ArrayList<String>(Arrays.asList("a", "b", "c","d"));Iterator<String> iter = list.iterator();while (iter.hasNext()) { String s = iter.next(); if (s.equals("a")) { iter.remove(); }}
next()方法需要在remove()方法之前被調(diào)用,在foreach循環(huán)里,編譯器會(huì)在刪除元素操作化調(diào)用next方法,這導(dǎo)致了ConcurrentModificationException異常。更多詳細(xì)信息,可以查看ArrayList.iterator()的源碼。
HashTable與HashMap
從算法的角度來(lái)講,HashTable是一種數(shù)據(jù)結(jié)構(gòu)名稱(chēng)。但是在Java中,這種數(shù)據(jù)結(jié)構(gòu)叫做HashMap。HashTable與HashMap的一個(gè)主要的區(qū)別是HashTable是同步的,所以,通常來(lái)說(shuō),你會(huì)使用HashMap,而不是Hashtable。
更多信息:HashMap vs. TreeMap vs. Hashtable vs. LinkedHashMap Top 10 questions about Map
使用集合原始類(lèi)型(raw type)
在Java中,原始類(lèi)型(raw type)和無(wú)界通配符類(lèi)型很容易讓人混淆。舉個(gè)Set的例子,Set是原始類(lèi)型,而Set是無(wú)界通配符類(lèi)型。
請(qǐng)看如下代碼,add方法使用了一個(gè)原始類(lèi)型的List作為入?yún)ⅲ?/p>
public static void add(List list, Object o){ list.add(o);}public static void main(String[] args){ List<String> list = new ArrayList<String>(); add(list, 10); String s = list.get(0);}
運(yùn)行以上代碼將會(huì)拋出異常:
Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Stringat ...
使用原始類(lèi)型集合非常危險(xiǎn),因?yàn)樗^(guò)了泛型類(lèi)型檢查,是不安全的。另外,Set, Set, 和Set這三個(gè)有很大的不同,具體請(qǐng)看:類(lèi)型擦除和Raw type vs. Unbounded wildcard。
訪問(wèn)級(jí)別
開(kāi)發(fā)人員經(jīng)常使用public修飾類(lèi)字段,雖然這很容易讓別人直接通過(guò)引用獲取該字段的值,但這是一個(gè)不好的設(shè)計(jì)。根據(jù)經(jīng)驗(yàn),應(yīng)該盡可能的降低成員屬性的訪問(wèn)級(jí)別。
相關(guān)閱讀:public, default, protected, and private
ArrayList和LinkedList
為什么開(kāi)發(fā)人員經(jīng)常使用ArrayList和LinkedList,卻不知道他們之間的區(qū)別,因?yàn)樗鼈兛雌饋?lái)很像。然而它們之間有著巨大的性能差異。簡(jiǎn)單的說(shuō),如果有大量的增加刪除操作并且沒(méi)有很多的隨機(jī)訪問(wèn)元素的操作,應(yīng)該首選LinkedList。
可變與不可變
不可變對(duì)象有很多優(yōu)點(diǎn),如簡(jiǎn)單、安全等。但是對(duì)于每個(gè)不同的值都需要一個(gè)單獨(dú)的對(duì)象,太多的對(duì)象會(huì)引起大量垃圾回收,因此在選擇可變與不可變的時(shí)候,需要有一個(gè)平衡。
通常,可變對(duì)象用于避免產(chǎn)生大量的中間對(duì)象,一個(gè)經(jīng)典的例子是大量字符串的拼接。如果你使用一個(gè)不可變對(duì)象,將會(huì)馬上產(chǎn)生大量符合垃圾回收標(biāo)準(zhǔn)的對(duì)象,這浪費(fèi)了CPU大量的時(shí)間和精力。使用可變對(duì)象是正確的解決方案(StringBuilder);
String result="";for(String s: arr){ result = result + s;}
另外,在有些其它情況下也是需要使用可變對(duì)象。例如往一個(gè)方法傳入一個(gè)可變對(duì)象,然后收集多種結(jié)果,而不需要
寫(xiě)太多的語(yǔ)法。另一個(gè)例子是排序和過(guò)濾:當(dāng)然,你可以寫(xiě)一個(gè)方法來(lái)接收原始的集合,并且返回一個(gè)排好序的集合,但是那樣對(duì)于大的集合就太浪費(fèi)了。
Why we need mutable classes?
父類(lèi)和子類(lèi)的構(gòu)造方法
之所以出現(xiàn)這個(gè)編譯錯(cuò)誤,是因?yàn)楦割?lèi)的默認(rèn)構(gòu)造方法未定義。在Java中,如果一個(gè)類(lèi)沒(méi)有定義構(gòu)造方法,編譯器會(huì)默認(rèn)插入一個(gè)無(wú)參數(shù)的構(gòu)造方法;但是如果一個(gè)構(gòu)造方法在父類(lèi)中已定義,在這種情況,編譯器是不會(huì)自動(dòng)插入一個(gè)默認(rèn)的無(wú)參構(gòu)造方法,這正是以上demo的情況;
對(duì)于子類(lèi)來(lái)說(shuō),不管是無(wú)參構(gòu)造方法還是有參構(gòu)造方法,都會(huì)默認(rèn)調(diào)用父類(lèi)的無(wú)參構(gòu)造方法;當(dāng)編譯器嘗試在子類(lèi)中往這兩個(gè)構(gòu)造方法插入super()方法時(shí),因?yàn)楦割?lèi)沒(méi)有一個(gè)默認(rèn)的無(wú)參構(gòu)造方法,所以編譯器報(bào)錯(cuò);
要修復(fù)這個(gè)錯(cuò)誤,很簡(jiǎn)單:
1、在父類(lèi)手動(dòng)定義一個(gè)無(wú)參構(gòu)造方法:
public Super(){ System.out.println("Super");}
2、移除父類(lèi)中自定義的構(gòu)造方法
3、在子類(lèi)中自己寫(xiě)上父類(lèi)構(gòu)造方法的調(diào)用;如super(value);
“”還是構(gòu)造方法
有兩種創(chuàng)建字符串的方式:
//1. use double quotesString x = "abc";//2. use constructorString y = new String("abc");
它們之間有什么區(qū)別呢?
以下代碼提供了一個(gè)快速回答:
String a = "abcd";String b = "abcd";System.out.println(a == b); // TrueSystem.out.println(a.equals(b)); // TrueString c = new String("abcd");String d = new String("abcd");System.out.println(c == d); // FalseSystem.out.println(c.equals(d)); // True
感謝各位的閱讀,以上就是“Java開(kāi)發(fā)人員有哪些最常犯的錯(cuò)誤”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Java開(kāi)發(fā)人員有哪些最常犯的錯(cuò)誤這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。