1. 程式人生 > 程式設計 >Java中List<T>和List<?>的區別詳解

Java中List<T>和List<?>的區別詳解

一、簡介

<T>在List、Set、Map中經常見到,用來限制Class中的引數型別,確保Class中引數的一致性。例如:List<String> list = new ArrayList<>();建立了一個內部引數是String型別的類,list中的操作物件都是String。<?>代表任意java型別,只有在不關心資料的具體型別下才使用萬用字元表示,但在一些情況下,需要將<?>傳入的資料進行強轉,但這樣不如直接傳入<T>。

另外除了<?>,還有<? extends T>上界萬用字元和<? super T>下界萬用字元。<? extends T> 表示傳入資料值需要是T型別或T的子類,<? suprt T>表示傳入資料值需要是T型別或T的超類。

一般來說,<?>主要用於變數上,<T>主要用於類或方法上。下圖中,list的元素型別為?,但往裡邊新增String時,會顯示出錯,因為list中的型別是一個未知的java型別,不屬於任何類,所以往裡邊新增資料時會出錯。但可以從list中取出資料,取出的資料型別為Object。

建議採用的順序是 List<T>、List<?>、List<Object>

二、程式碼例項

Java中List<T>和List<?>的區別詳解

List<?>是隻讀型別的,不能進行增加、修改操作。

Java中List<T>和List<?>的區別詳解

Java中List<T>和List<?>的區別詳解

<?>的各種坑

但 List<?>這個寫法非常坑。因為,這時候萬用字元會捕獲具體的String型別,但編譯器不叫它String,而是起個臨時的代號,比如”CAP#1“。所以以後再也不能往list裡存任何元素,包括String。唯一能存的就是null。

List<?> list = new ArrayList<String>();
 
list.add("hello"); //ERROR
list.add(111); //ERROR
 
//argument mismatch; String cannot be converted to CAP#1
//argument mismatch; int cannot be converted to CAP#1

另外如果拿List<?>做引數,也會有奇妙的事情發生。還是剛才Box<T>的例子,有get()和set()兩個方法,一個存,一個取。

class Box<T>{
 private List<T> item;
 public List<T> get(){return item;}
 public void set(List<T> t){item=t;}
 //把item取出來,再放回去
 public void getSet(Box<?> box){box.set(box.get());} //ERROR
}

新的getSet()方法,只是把item先用get()方法讀出來,然後再用set()方法存回去。按理說不可能有問題。實際執行卻會報錯。

error: incompatible types: Object cannot be converted to CAP#1

原因和前面一樣,萬用字元box<?>.set()的引數型別被編譯器捕獲,命名為CAP#1,和box<?>.get()返回的Object物件無法匹配。

解決方法,是要給getSet()方法寫一個 輔助函式,具體原理可以去查《Java核心技術-卷1》,泛型這章,或者《Java程式設計思想》。都有講。

class Box<T>{
 private List<T> item;
 public List<T> get(){return item;}
 public void set(List<T> t){item=t;}
 //helper()函式輔助getSet()方法存取元素
 public void getSet(Box<?> box){helper(box);}
 public <V> void helper(Box<V> box){box.set(box.get());}
}

到此這篇關於Java中List<T>和List<?>的區別詳解的文章就介紹到這了,更多相關List<T>和List<?>內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!