溫馨提示×

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

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

通過(guò)代碼理解java泛型

發(fā)布時(shí)間:2020-09-29 14:49:26 來(lái)源:腳本之家 閱讀:117 作者:陳灬大灬海 欄目:編程語(yǔ)言

泛型數(shù)據(jù)java基礎(chǔ),但真正理解需要悉心品嘗。畢竟在工作中用到的是在是太多了。

不要以為new ArrayList<>這就是泛型,這只能屬于會(huì)使用。

在工作中,相對(duì)于現(xiàn)有的項(xiàng)目源碼的數(shù)據(jù)庫(kù)操作層,無(wú)論是mybatis,hibernate或者是自己封裝的baseModel層,都會(huì)使用到泛型。

以及<? extends T> 和 <? super T>這個(gè)屌東西。

泛型使用情況分為三類

1. 泛型類。

2. 泛型方法。

3. 泛型接口。

出于規(guī)范的目的,Java 還是建議我們用單個(gè)大寫字母來(lái)代表類型參數(shù)。常見(jiàn)的如:

1. T 代表一般的任何類。

2. E 代表 Element 的意思,或者 Exception 異常的意思。

3. K 代表 Key 的意思。

4. V 代表 Value 的意思,通常與 K 一起配合使用。

5. S 代表 Subtype 的意思,文章后面部分會(huì)講解示意。

最直接的一段代碼。

List<String> l1 = new ArrayList<String>();
  List<Integer> l2 = new ArrayList<Integer>();
  System.out.println(l1.getClass() == l2.getClass());

打印的判斷為TRUE,因?yàn)榉盒托畔⒈徊脸恕?/p>

泛型擦除實(shí)例。

List<String> listErasure = new ArrayList<String>() {
   // 直接初始化,這也是一種方式。直接傳入一個(gè)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類型,但是卻可以通過(guò)反射對(duì)其進(jìn)行Integer類型的操作。

可見(jiàn)泛型只是在編譯期間有效。

<?> 代表著類型未知

<? extends T> 和 <? super T>這個(gè)東西經(jīng)常見(jiàn)到,但是并沒(méi)有字面意義那么簡(jiǎn)單。

通配符有 3 種形式。

  1. <?> 被稱作無(wú)限定的通配符。
  2. <? extends T> 被稱作有上限的通配符。
  3. <? super T> 被稱作有下限的通配符。
class A{}
class B extends A{}
class C extends B{}
List<? extends B> listExtends = new ArrayList<B>();
//  listExtends.add(new A()); 全部編譯錯(cuò)誤。因?yàn)槭褂玫氖莈xtends,喪失了寫的操作能力。跟f3方法一樣,是未知類型,只是確定了里面對(duì)象的范圍。是B的子類。
//  listExtends.add(new B());
//  listExtends.add(new C());
  
  // 能進(jìn)行對(duì)B以及B的子類操作。這是super的神奇之處。
  List<? super B> listSuper = new ArrayList<B>();
//  listSuper.add(new A());//會(huì)編譯錯(cuò)誤。
  listSuper.add(new B());
  listSuper.add(new C());

以及方法泛型的返回

泛型作為參數(shù)的傳遞。

public static <TTT>TTT f1(TTT t) {
  return t;
 }
 
 // 傳遞指定的A類型,對(duì)應(yīng)的list可以進(jìn)行對(duì)應(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)行跟?無(wú)關(guān)的操作,類似于size方法,增加代碼的可讀性。
  System.out.println(list.size());
 }
 
 public static void f4(List<? extends B> listExtends) {
//  listExtends.add(new B());//不能進(jìn)行寫做操,因?yàn)槭?#63;,增加了可讀性。
  System.out.println(listExtends.size());
 }

測(cè)試代碼,很全面

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類型,對(duì)應(yīng)的list可以進(jìn)行對(duì)應(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)行跟?無(wú)關(guān)的操作,類似于size方法,增加代碼的可讀性。
  System.out.println(list.size());
 }
 public static void f4(List<? extends B> listExtends) {
//  listExtends.add(new B());//不能進(jìn)行寫做操,因?yàn)槭?#63;,增加了可讀性。
  System.out.println(listExtends.size());
 }
 public static void main(String[] args) throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException, InstantiationException {
  // 打印的判斷為TRUE,因?yàn)榉盒托畔⒈徊脸恕?  List<String> l1 = new ArrayList<String>();
  List<Integer> l2 = new ArrayList<Integer>();
  System.out.println(l1.getClass() == l2.getClass());
  // 泛型擦除實(shí)例。
  List<String> listErasure = new ArrayList<String>() {
   // 直接初始化,這也是一種方式。直接傳入一個(gè)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); // 不對(duì)f3方法進(jìn)行任何操作,是1.
  f2(listA); // 2對(duì)應(yīng)的方法實(shí)現(xiàn)還進(jìn)行了一次插入操作。
  f3(listA); // static ,對(duì)應(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()); 全部編譯錯(cuò)誤。因?yàn)槭褂玫氖莈xtends,喪失了寫的操作能力。跟f3方法一樣,是未知類型,只是確定了里面對(duì)象的范圍。是B的子類。
//  listExtends.add(new B());
//  listExtends.add(new C());
  
  // 能進(jìn)行對(duì)B以及B的子類操作。這是super的神奇之處。
  List<? super B> listSuper = new ArrayList<B>();
//  listSuper.add(new A());//會(huì)編譯錯(cuò)誤。
  listSuper.add(new B());
  listSuper.add(new C());
  
  
  // 沒(méi)毛病。
  List<B> listBS = new ArrayList<B>(); 
  listBS.add(new B());
  f4(listBS); 
 }
}

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

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

AI