大战熟女丰满人妻av-荡女精品导航-岛国aaaa级午夜福利片-岛国av动作片在线观看-岛国av无码免费无禁网站-岛国大片激情做爰视频

專注Java教育14年 全國咨詢/投訴熱線:400-8080-105
動力節(jié)點(diǎn)LOGO圖
始于2009,口口相傳的Java黃埔軍校
首頁 學(xué)習(xí)攻略 Java程序員常犯的10個錯誤

Java程序員常犯的10個錯誤

更新時(shí)間:2019-07-31 19:47:41 來源:動力節(jié)點(diǎn) 瀏覽2484次



人非圣賢,孰能無過。都說Java語言是一門簡單的編程語言,基于C++演化而來,剔除了很多C++中的復(fù)雜特性,但這并不能保證Java程序員不會犯錯。那么對于廣大的Java程序員來說,它們最常犯的10個錯誤是什么呢?本文是動力節(jié)點(diǎn)java學(xué)院小編總結(jié)的Java程序員最常犯的10大錯誤,可以有效的幫助Java后來者少走彎路,少加班,并寫出更健壯的應(yīng)用程序。



1、訪問級別


開發(fā)人員經(jīng)常使用public修飾類字段,雖然這很容易讓別人直接通過引用獲取該字段的值,但這是一個不好的設(shè)計(jì)。根據(jù)經(jīng)驗(yàn),應(yīng)該盡可能的降低成員屬性的訪問級別。


相關(guān)閱讀:public, default, protected, and private



2、父類和子類的構(gòu)造方法


之所以出現(xiàn)這個編譯錯誤,是因?yàn)楦割惖哪J(rèn)構(gòu)造方法未定義。在Java中,如果一個類沒有定義構(gòu)造方法,編譯器會默認(rèn)插入一個無參數(shù)的構(gòu)造方法;但是如果一個構(gòu)造方法在父類中已定義,在這種情況,編譯器是不會自動插入一個默認(rèn)的無參構(gòu)造方法,這正是以上demo的情況;


對于子類來說,不管是無參構(gòu)造方法還是有參構(gòu)造方法,都會默認(rèn)調(diào)用父類的無參構(gòu)造方法;當(dāng)編譯器嘗試在子類中往這兩個構(gòu)造方法插入super()方法時(shí),因?yàn)楦割悰]有一個默認(rèn)的無參構(gòu)造方法,所以編譯器報(bào)錯;


要修復(fù)這個錯誤,很簡單:


1、在父類手動定義一個無參構(gòu)造方法:


public Super(){ System.out.println("Super");}


2、移除父類中自定義的構(gòu)造方法


3、在子類中自己寫上父類構(gòu)造方法的調(diào)用;如super(value);



3、使用集合原始類型(raw type)


在Java中,原始類型(raw type)和無界通配符類型很容易讓人混淆。舉個Set的例子,Set是原始類型,而Set是無界通配符類型。


請看如下代碼,add方法使用了一個原始類型的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)行以上代碼將會拋出異常:


Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Stringat ...


使用原始類型集合非常危險(xiǎn),因?yàn)樗^了泛型類型檢查,是不安全的。另外,Set, Set, 和Set這三個有很大的不同,具體請看:類型擦除和Raw type vs. Unbounded wildcard。



4、“”還是構(gòu)造方法


有兩種創(chuàng)建字符串的方式:


//1. use double quotesString x = "abc";//2. use constructorString y = new String("abc");


它們之間有什么區(qū)別呢?


以下代碼提供了一個快速回答:


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


更多關(guān)于它們內(nèi)存分配的信息,請參考Create Java String Using ” ” or Constructor??



5、Array轉(zhuǎn)ArrayList


當(dāng)需要把Array轉(zhuǎn)成ArrayList的時(shí)候,開發(fā)人員經(jīng)常這樣做:


List<String> list = Arrays.asList(arr);


Arrays.asList()會返回一個ArrayList,但是要特別注意,這個ArrayList是Arrays類的靜態(tài)內(nèi)部類,并不是java.util.ArrayList類。java.util.Arrays.ArrayList類實(shí)現(xiàn)了set(), get(),contains()方法,但是并沒有實(shí)現(xiàn)增加元素的方法(事實(shí)上是可以調(diào)用add方法,但是沒有具體實(shí)現(xiàn),僅僅拋出UnsupportedOperationException異常),因此它的大小也是固定不變的。為了創(chuàng)建一個真正的java.util.ArrayList,你應(yīng)該這樣做:


ArrayList<String> arrayList = new ArrayList<String>(Arrays.asList(arr));


ArrayList的構(gòu)造方法可以接收一個Collection類型,而java.util.Arrays.ArrayList已經(jīng)實(shí)現(xiàn)了該接口。



6、可變與不可變


不可變對象有很多優(yōu)點(diǎn),如簡單、安全等。但是對于每個不同的值都需要一個單獨(dú)的對象,太多的對象會引起大量垃圾回收,因此在選擇可變與不可變的時(shí)候,需要有一個平衡。


通常,可變對象用于避免產(chǎn)生大量的中間對象,一個經(jīng)典的例子是大量字符串的拼接。如果你使用一個不可變對象,將會馬上產(chǎn)生大量符合垃圾回收標(biāo)準(zhǔn)的對象,這浪費(fèi)了CPU大量的時(shí)間和精力。使用可變對象是正確的解決方案(StringBuilder);


String result="";for(String s: arr){ result = result + s;}


另外,在有些其它情況下也是需要使用可變對象。例如往一個方法傳入一個可變對象,然后收集多種結(jié)果,而不需要寫太多的語法。另一個例子是排序和過濾:當(dāng)然,你可以寫一個方法來接收原始的集合,并且返回一個排好序的集合,但是那樣對于大的集合就太浪費(fèi)了。


更多閱讀:Why String is Immutable??


Why we need mutable classes?



7、判斷一個數(shù)組是否包含某個值


開發(fā)人員經(jīng)常這樣做:


Set<String> set = new HashSet<String>(Arrays.asList(arr));return set.contains(targetValue);


以上代碼可以正常工作,但是沒有必要將其轉(zhuǎn)換成set集合,將一個List轉(zhuǎn)成Set需要額外的時(shí)間,其實(shí)我們可以簡單的使用如下方法即可:


Arrays.asList(arr).contains(targetValue);


或者


for(String s: arr){ if(s.equals(targetValue)) return true;}return false;


第一種方法可讀性更強(qiáng)。



8、在循環(huán)內(nèi)部刪除List中的一個元素


考慮如下代碼,在迭代期間刪除元素:


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é)果打印:


[b, d]


在上面這個方法中有一系列的問題,當(dāng)一個元素被刪除的時(shí)候,list大小減小,然后原先索引指向了其它元素。所以如果你想在循環(huán)里通過索引來刪除多個元素,將不會正確工作。


你也許知道使用迭代器是在循環(huán)里刪除元素的正確方式,或許你也知道foreach循環(huán)跟迭代器很類似,但事實(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異常。


然而接下來的代碼卻是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)里,編譯器會在刪除元素操作化調(diào)用next方法,這導(dǎo)致了ConcurrentModificationException異常。更多詳細(xì)信息,可以查看ArrayList.iterator()的源碼。



9、HashTable與HashMap


從算法的角度來講,HashTable是一種數(shù)據(jù)結(jié)構(gòu)名稱。但是在Java中,這種數(shù)據(jù)結(jié)構(gòu)叫做HashMap。HashTable與HashMap的一個主要的區(qū)別是HashTable是同步的,所以,通常來說,你會使用HashMap,而不是Hashtable。


更多信息:HashMap vs. TreeMap vs. Hashtable vs. LinkedHashMap Top 10 questions about Map



10、ArrayList和LinkedList


為什么開發(fā)人員經(jīng)常使用ArrayList和LinkedList,卻不知道他們之間的區(qū)別,因?yàn)樗鼈兛雌饋砗芟瘛H欢鼈冎g有著巨大的性能差異。簡單的說,如果有大量的增加刪除操作并且沒有很多的隨機(jī)訪問元素的操作,應(yīng)該首選LinkedList。




提交申請后,顧問老師會電話與您溝通安排學(xué)習(xí)

免費(fèi)課程推薦 >>
技術(shù)文檔推薦 >>
主站蜘蛛池模板: 老妇激情毛片免费 | 成人在线免费小视频 | 亚洲rv国产rv日本rv | 国产亚洲精品网站 | 欧美精品在欧美一区二区 | 四虎影院大全 | 久久国产高清 | 热热涩热热狠狠色香蕉综合 | 欧美三级欧美成人高清www | 偷亚洲偷国产欧美高清 | 日韩欧美一卡二区 | 日韩久久久精品中文字幕 | 2021国产精品系列一区二区 | 性欧美成人依依影院 | 四库国产精品成人 | 色婷婷久久合月综 | 67194在线午夜亚洲 | 日本xoxoxo在线播放 | 日本亚洲欧洲免费无码 | 国产乱人伦av在线a 国产乱人伦精品一区二区 国产乱人伦偷精品视频不卡 | 黄色资源在线观看 | 亚洲欧美日韩中文高清一 | 久久国产精品999 | 日本高清一级做a爱过程免费视频 | 99久久99热精品免费观看国产 | 成人网中文字幕色 | 亚洲狼人综合干 | 性欧美视频a毛片在线播放 性欧美视频在线观看 | 久久这里只有精品1 | 再猛点深使劲爽日本免费视频 | 成人a毛片手机免费播放 | 热久久国产欧美一区二区精品 | 四虎最新地址 | 欧美aaa大片 | 草免费视频 | 五月婷婷久久综合 | 亚洲精品国产精品一区二区 | 国产自产拍精品视频免费看 | 狠狠色伊人亚洲综合成人 | 神马影院我不卡888 神马影院我不卡手机 | 自拍偷拍三级 |