溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

總結(jié)軟件開發(fā)過程中最常犯的10個問題

發(fā)布時間:2020-06-13 04:05:05 來源:網(wǎng)絡(luò) 閱讀:380 作者:愛碼仕i 欄目:編程語言

總結(jié)軟件開發(fā)過程中最常犯的10個問題

先總結(jié)一下Top10吧

  1. Array轉(zhuǎn)ArrayList
  2. 判斷一個數(shù)組是否包含某個值
  3. 在循環(huán)內(nèi)部刪除List中的一個元素
  4. HashTable與HashMap
  5. 使用集合原始類型(raw type)
  6. 訪問級別
  7. ArrayList和LinkedList
  8. 可變與不可變
  9. 父類和子類的構(gòu)造方法
  10. “”還是構(gòu)造方法

這個列表總結(jié)了10個Java開發(fā)人員最常犯的錯誤

1、Array轉(zhuǎn)ArrayList

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

List list = Arrays.asList(arr);

Arrays.asList()會返回一個ArrayList,但是要特別注意,這個ArrayListArrays類的靜態(tài)內(nèi)部類,并不是java.util.ArrayList類。

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

ArrayList arrayList = new ArrayList(Arrays.asList(arr));

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

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

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

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

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

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

或者

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

第一種方法可讀性更強。

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

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

ArrayList list = new ArrayList(Arrays.asList("a", "b", "c","d"));
for (int i = 0; i
list.remove(i);
}
System.out.println(list);

結(jié)果打?。?code>[b, d]

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

你也許知道使用迭代器是在循環(huán)里刪除元素的正確方式,或許你也知道foreach循環(huán)跟迭代器很類似,但事實情況卻不是這樣,如下代碼:

ArrayList list = new ArrayList(Arrays.asList("a", "b", "c","d"));
for (String s : list) {
    if (s.equals("a"))
    list.remove(s);
}

將拋出ConcurrentModificationException異常。

然而接下來的代碼卻是OK的:

ArrayList list = new ArrayList(Arrays.asList("a", "b", "c","d"));
Iterator 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()的源碼。

4、HashTable與HashMap

從算法的角度來講,HashTable是一種數(shù)據(jù)結(jié)構(gòu)名稱。但是在Java中,這種數(shù)據(jù)結(jié)構(gòu)叫做HashMap

HashTableHashMap的一個主要的區(qū)別是HashTable是同步的,所以,通常來說,你會使用HashMap,而不是Hashtable。

5、使用集合原始類型(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 list = new ArrayList();
    add(list, 10);
    String s = list.get(0);
}

運行以上代碼將會拋出異常:

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

使用原始類型集合非常危險,因為它跳過了泛型類型檢查,是不安全的。另外,Set, SetSet這三個有很大的不同。

6、訪問級別

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

7、ArrayList和LinkedList

為什么開發(fā)人員經(jīng)常使用ArrayListLinkedList,卻不知道他們之間的區(qū)別,因為它們看起來很像。然而它們之間有著巨大的性能差異。簡單的說,如果有大量的增加刪除操作并且沒有很多的隨機訪問元素的操作,應(yīng)該首選LinkedList

8、可變與不可變

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

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

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

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

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

class Super{
    string s;
    public Super(String s){
        this.s=s;
    }
}
public class Sub extends Super{
    int x=200;
    public Sub(String s){
    }
    public Sub(){
        System.out.println("Sub");
    }
    public static void main(String[] args){
        Sub s=new Sub();
    }
}

之所以出現(xiàn)這個編譯錯誤,是因為父類的默認(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()方法時,因為父類沒有一個默認(rèn)的無參構(gòu)造方法,所以編譯器報錯;

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

  1. 在父類手動定義一個無參構(gòu)造方法:
    public Super(){
    System.out.println("Super");
    }
  2. 移除父類中自定義的構(gòu)造方法
  3. 在子類中自己寫上父類構(gòu)造方法的調(diào)用;如super(value)

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

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

String x = "abc";
String y = new String("abc");

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

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

String a = "abcd";
String b = "abcd";
System.out.println(a == b);
// True
System.out.println(a.equals(b));
// True
String c = new String("abcd");
String d = new String("abcd");
System.out.println(c == d);
// False
System.out.println(c.equals(d));
// True

寫在最后

  • 第一:看完點贊,感謝您的認(rèn)可;
  • ...
  • 第二:隨手轉(zhuǎn)發(fā),分享知識,讓更多人學(xué)習(xí)到;
  • ...
  • 第三:記得點關(guān)注,每天更新的?。?!
  • ...

總結(jié)軟件開發(fā)過程中最常犯的10個問題

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI