通過代碼理解java泛型
泛型數(shù)據(jù)java基礎(chǔ),但真正理解需要悉心品嘗。畢竟在工作中用到的是在是太多了。
不要以為new ArrayList<>這就是泛型,這只能屬于會使用。
在工作中,相對于現(xiàn)有的項目源碼的數(shù)據(jù)庫操作層,無論是mybatis,hibernate或者是自己封裝的baseModel層,都會使用到泛型。
以及<? extends T> 和 <? super T>這個屌東西。
泛型使用情況分為三類
1. 泛型類。
2. 泛型方法。
3. 泛型接口。
出于規(guī)范的目的,Java 還是建議我們用單個大寫字母來代表類型參數(shù)。常見的如:
1. T 代表一般的任何類。
2. E 代表 Element 的意思,或者 Exception 異常的意思。
3. K 代表 Key 的意思。
4. V 代表 Value 的意思,通常與 K 一起配合使用。
5. S 代表 Subtype 的意思,文章后面部分會講解示意。
最直接的一段代碼。
List<String> l1 = new ArrayList<String>(); List<Integer> l2 = new ArrayList<Integer>(); System.out.println(l1.getClass() == l2.getClass());
打印的判斷為TRUE,因為泛型信息被擦除了。
泛型擦除實例。
List<String> listErasure = new ArrayList<String>() {
// 直接初始化,這也是一種方式。直接傳入一個collection。
{add("aaa");add("bbb");}
};
listErasure.add("ccc");
Class<? extends List> class1 = listErasure.getClass();
Method method = class1.getMethod("add",Object.class);
method.invoke(listErasure, 123);
System.out.println(listErasure)
輸出結(jié)果 [aaa, bbb, ccc, 123]
明明是接收String類型,但是卻可以通過反射對其進(jìn)行Integer類型的操作。
可見泛型只是在編譯期間有效。
<?> 代表著類型未知
<? extends T> 和 <? super T>這個東西經(jīng)常見到,但是并沒有字面意義那么簡單。
通配符有 3 種形式。
- <?> 被稱作無限定的通配符。
- <? extends T> 被稱作有上限的通配符。
- <? super T> 被稱作有下限的通配符。
class A{}
class B extends A{}
class C extends B{}
List<? extends B> listExtends = new ArrayList<B>(); // listExtends.add(new A()); 全部編譯錯誤。因為使用的是extends,喪失了寫的操作能力。跟f3方法一樣,是未知類型,只是確定了里面對象的范圍。是B的子類。 // listExtends.add(new B()); // listExtends.add(new C()); // 能進(jìn)行對B以及B的子類操作。這是super的神奇之處。 List<? super B> listSuper = new ArrayList<B>(); // listSuper.add(new A());//會編譯錯誤。 listSuper.add(new B()); listSuper.add(new C());
以及方法泛型的返回
泛型作為參數(shù)的傳遞。
public static <TTT>TTT f1(TTT t) {
return t;
}
// 傳遞指定的A類型,對應(yīng)的list可以進(jìn)行對應(yīng)的list應(yīng)有的方法。
public static void f2(List<A> list) {
list.add(new A());
System.out.println(list.size());
}
public static void f3(List<?> list) {
// list.add(new A()); //當(dāng)傳入的是?通配符的話表示只能進(jìn)行跟?無關(guān)的操作,類似于size方法,增加代碼的可讀性。
System.out.println(list.size());
}
public static void f4(List<? extends B> listExtends) {
// listExtends.add(new B());//不能進(jìn)行寫做操,因為是?,增加了可讀性。
System.out.println(listExtends.size());
}
測試代碼,很全面
package com.javaSE.fanxing;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
class A{}
class B extends A{}
class C extends B{}
public class Demo<T,TT> {
T value;
TT value2;
public TT getValue2() {
return value2;
}
public void setValue2(TT value2) {
this.value2 = value2;
}
public T getValue() {
return value;
}
public void setValue(T value) {
this.value = value;
}
public static <TTT>TTT f1(TTT t) {
return t;
}
// 傳遞指定的A類型,對應(yīng)的list可以進(jìn)行對應(yīng)的list應(yīng)有的方法。
public static void f2(List<A> list) {
list.add(new A());
System.out.println(list.size());
}
public static void f3(List<?> list) {
// list.add(new A()); //當(dāng)傳入的是?通配符的話表示只能進(jìn)行跟?無關(guān)的操作,類似于size方法,增加代碼的可讀性。
System.out.println(list.size());
}
public static void f4(List<? extends B> listExtends) {
// listExtends.add(new B());//不能進(jìn)行寫做操,因為是?,增加了可讀性。
System.out.println(listExtends.size());
}
public static void main(String[] args) throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException, InstantiationException {
// 打印的判斷為TRUE,因為泛型信息被擦除了。
List<String> l1 = new ArrayList<String>();
List<Integer> l2 = new ArrayList<Integer>();
System.out.println(l1.getClass() == l2.getClass());
// 泛型擦除實例。
List<String> listErasure = new ArrayList<String>() {
// 直接初始化,這也是一種方式。直接傳入一個collection。
{add("aaa");add("bbb");}
};
listErasure.add("ccc");
Class<? extends List> class1 = listErasure.getClass();
Method method = class1.getMethod("add",Object.class);
method.invoke(listErasure, 123);
System.out.println(listErasure);
Demo<String,Integer> demo = new Demo<String,Integer>();
demo.setValue("string");
System.out.println(demo.getValue());
Demo<Integer,String> demo2 = new Demo<Integer,String>();
demo2.setValue(100);
System.out.println(demo2.getValue());
System.out.println(f1(123));
// List<A> listA = new ArrayList<A>();
// List<B> listB = listA;//new ArrayList<B>();雖然B是A的子類,并不代表泛型之間也具備繼承關(guān)系。
ArrayList<A> listA = new ArrayList<A>();
listA.add(new A());
f3(listA); // 不對f3方法進(jìn)行任何操作,是1.
f2(listA); // 2對應(yīng)的方法實現(xiàn)還進(jìn)行了一次插入操作。
f3(listA); // static ,對應(yīng)的listA的集合數(shù)量是引用值。
ArrayList<B> listB = new ArrayList<B>();
listB.add(new B());
f3(listB); // f3方法傳遞的是通配符?,不能進(jìn)行add操作。
// <? extends T> 和 <? super T>
List<? extends B> listExtends = new ArrayList<B>();
// listExtends.add(new A()); 全部編譯錯誤。因為使用的是extends,喪失了寫的操作能力。跟f3方法一樣,是未知類型,只是確定了里面對象的范圍。是B的子類。
// listExtends.add(new B());
// listExtends.add(new C());
// 能進(jìn)行對B以及B的子類操作。這是super的神奇之處。
List<? super B> listSuper = new ArrayList<B>();
// listSuper.add(new A());//會編譯錯誤。
listSuper.add(new B());
listSuper.add(new C());
// 沒毛病。
List<B> listBS = new ArrayList<B>();
listBS.add(new B());
f4(listBS);
}
}
相關(guān)文章
java wait()/notify() 實現(xiàn)生產(chǎn)者消費(fèi)者模式詳解
這篇文章主要介紹了java wait()/notify() 實現(xiàn)生產(chǎn)者消費(fèi)者模式詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
用IDEA創(chuàng)建SpringBoot項目的詳細(xì)步驟記錄
Idea有著非常簡便的Spring Boot新建過程,同時依靠pom自動下載依賴,下面這篇文章主要給大家介紹了關(guān)于用IDEA創(chuàng)建SpringBoot項目的詳細(xì)步驟,文中通過圖文介紹的非常詳細(xì),需要的朋友可以參考下2022-08-08
在MyBatis中實現(xiàn)一對多查詢和多對一查詢的方式詳解(各兩種方式)
今天通過兩種方法分別給大家介紹在MyBatis中實現(xiàn)一對多查詢和多對一查詢的方式,每種方式通過實例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2022-01-01
springboot2.2.2集成dubbo的實現(xiàn)方法
這篇文章主要介紹了springboot2.2.2集成dubbo的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
MyBatis加載映射文件和動態(tài)代理的實現(xiàn)
本文主要介紹了MyBatis加載映射文件和動態(tài)代理的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05
springboot結(jié)合mysql主從來實現(xiàn)讀寫分離的方法示例
這篇文章主要介紹了springboot結(jié)合mysql主從來實現(xiàn)讀寫分離的方法示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
以實例講解Objective-C中的KVO與KVC機(jī)制
這篇文章主要介紹了以實例講解Objective-C中的KVO與KVC機(jī)制,即Key-Value-Observing與Key-Value-Coding,需要的朋友可以參考下2015-09-09

