溫馨提示×

溫馨提示×

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

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

Java 泛型在哪些情況下無法使用

發(fā)布時間:2021-11-09 18:27:28 來源:億速云 閱讀:142 作者:柒染 欄目:大數(shù)據(jù)

今天就跟大家聊聊有關Java 泛型在哪些情況下無法使用,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

1. 前言

Java 1.5引入了泛型來保證類型安全,防止在運行時發(fā)生類型轉(zhuǎn)換異常,讓類型參數(shù)化,提高了代碼的可讀性和重用率。但是有些情況下泛型也是不允許使用的,今天就總結(jié)一下編碼中不能使用泛型的一些場景。

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

以下寫法是錯誤的:

// 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. 泛型無法作為靜態(tài)變量類型

Java中的靜態(tài)類型隨著類加載而實例化,此時泛型的具體類型并沒有聲明。同時因為靜態(tài)變量作為所有對象的共享變量,只有類實例化或者方法調(diào)用時才能確定其類型。如果是泛型類型將無法確定其類型。同樣在類上聲明的泛型也無法作為返回值類型出現(xiàn)在類的靜態(tài)方法中,下面的寫法也是錯誤的:

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

5. 無法進行 instanceof 判斷

Java中的泛型是偽泛型,在編譯期會被擦除,運行的字節(jié)碼中不存在泛型,所以下面的判斷條件無法進行:

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

但是泛型的無界通配符<?>可以進行instanceof判斷,你仔細想想為什么。

 

6. 無法創(chuàng)建參數(shù)化類型的數(shù)組

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

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

但是加上了泛型就編譯不通過了:

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

如果不這么規(guī)定將引發(fā)以下邏輯錯誤:

// 如果上面的成立,則下面的也應該成立
Object[] stringLists = new List<String>[];
// 那么我們可以放入 字符串 List
stringLists[0] = new ArrayList<String>();
// 放入 Integer list
stringLists[1] = new ArrayList<Integer>();
// 這顯然不合理
   

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

下面的兩種寫法將引發(fā)編譯錯誤:

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

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

如果成立將出現(xiàn):

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

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

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

8. 泛型擦除后相同參數(shù)簽名的方法不能重載

由于泛型擦除的原因,以下的不視為方法的重載且無法編譯 :

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

看完上述內(nèi)容,你們對Java 泛型在哪些情況下無法使用有進一步的了解嗎?如果還想了解更多知識或者相關內(nèi)容,請關注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細節(jié)

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

AI