爲何公司新來的阿里P8大佬說:這幾種狀況下不能使用Java泛型

最新互聯網大廠面試真題、Java程序員面試策略(面試前的準備、面試中的技巧)請訪問GitHubgit

1. 前言

Java 1.5 引入了泛型來保證類型安全,防止在運行時發生類型轉換異常,讓類型參數化,提升了代碼的可讀性和重用率。可是有些狀況下泛型也是不容許使用的,今天就總結一下編碼中不能使用泛型的一些場景。程序員

2. 基本類型沒法直接使用泛型

如下寫法是錯誤的:github

// error 
Map<int,char> wrong= new HashMap<>()

基本類型是不可以做爲泛型類型的,須要使用它們對應的包裝類。面試

// OK
Map<Integer,Character> wrong= new HashMap<>()

3. 泛型類型沒法被直接實例化

泛型類型能夠理解爲一個抽象類型,只是表明了類型的抽象,所以咱們不能直接實例化它,下面的作法也是錯誤的:數組

public <E> E first(List<E> list){
     // error 
        E e = new E();
       return list.get(0);   
 }

4. 泛型沒法做爲靜態變量類型

Java 中的靜態類型隨着類加載而實例化,此時泛型的具體類型並無聲明。同時由於靜態變量做爲全部對象的共享變量,只有類實例化或者方法調用時才能肯定其類型。若是是泛型類型將沒法肯定其類型。一樣在類上聲明的泛型也沒法做爲返回值類型出如今類的靜態方法中,下面的寫法也是錯誤的:安全

public class Generic<T>{
    // 不能將類聲明的泛型類型做爲靜態變量
    public static T t;
    // 也不能將類聲明的泛型類型做爲 靜態方法的返回值
    public  static  T  rtval(List<T> list){
        return list.get(0);
    }
}

5. 沒法進行 instanceof 判斷

Java 中的泛型是僞泛型,在編譯期會被擦除,運行的字節碼中不存在泛型,因此下面的判斷條件沒法進行:ide

public static <E> void wrong(List<E> list) {
    // error 
    if (list instanceof ArrayList<Integer>) {   
    }
}

可是泛型的***通配符&lt;?&gt;能夠進行instanceof判斷,你仔細想一想爲何。編碼

6. 沒法建立參數化類型的數組

首先下面這種寫法是對的:code

// OK
List[] arrayOfLists = new List[2];

可是加上了泛型就編譯不經過了:對象

//error
List<Integer>[] arrayOfLists = new List<Integer>[2];

若是不這麼規定將引起如下邏輯錯誤:

// 若是上面的成立,則下面的也應該成立
Object[] stringLists = new List<String>[];  
// 那麼咱們能夠放入 字符串 List
stringLists[0] = new ArrayList<String>();   
// 放入 Integer list
stringLists[1] = new ArrayList<Integer>();
// 這顯然不合理

7. 不能直接或者間接擴展Throwable

下面的兩種寫法將引起編譯錯誤:

//  不能間接地擴展 Throwable   
class IndirectException<T> extends Exception {}     

//  不能直接地擴展 Throwable  
class DirectException<T> extends Throwable {}

若是成立將出現:

try {
        // ...
    } catch (T e) {   
        // 類型不肯定  沒法處理具體的異常邏輯
    }

你如何才能對異常進行具體的處理,這顯然不便於精確的異常處理邏輯。可是你能夠拋出一個 不肯定的異常,可是一樣不能在靜態方法中使用類聲明的泛型:

class Parser<T extends Exception> {
   // 這樣是對的
    public void okThrow(File file) throws T {      
        // ...
    }
    // 靜態方法不能出現類聲明的泛型類型做爲返回值和異常
    public static void wrongThrow(File file) throws T {      
    }
}

8. 泛型擦除後相同參數簽名的方法不能重載

因爲泛型擦除的緣由,如下的不視爲方法的重載且沒法編譯 :

public class NoReload {
    public void sets(Set<String> strSet) { }
    public void sets(Set<Integer> intSet) { }
}

9. 總結

今天總結了 Java 泛型的一些使用誤區,雖然日常 IDE 的提示會告訴咱們,可是這也是咱們常常會忽略的一些知識點。

相關文章
相關標籤/搜索