成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

聊一聊Java 泛型全解

開發 后端
對于java的泛型我一直屬于一知半解的,平常真心用的不多。直到閱讀《Effect Java》,看到很多平常不了解的用法,才下定決心,需要系統的學習,并且記錄下來。

 對于java的泛型我一直屬于一知半解的,平常真心用的不多。直到閱讀《Effect Java》,看到很多平常不了解的用法,才下定決心,需要系統的學習,并且記錄下來。

[[275234]]

1、泛型的概述:

1.1 泛型的由來

根據《Java編程思想》中的描述,泛型出現的動機:

有很多原因促成了泛型的出現,而最引人注意的一個原因,就是為了創建容器類。

泛型的思想很早就存在,如C++中的模板(Templates)。模板的精神:參數化類型

1.2 基本概述

  • 泛型的本質就是"參數化類型"。一提到參數,最熟悉的就是定義方法的時候需要形參,調用方法的時候,需要傳遞實參。那"參數化類型"就是將原來具體的類型參數化
  • 泛型的出現避免了強轉的操作,在編譯器完成類型轉化,也就避免了運行的錯誤。

1.3 泛型的目的

Java泛型也是一種語法糖,在編譯階段完成類型的轉換的工作,避免在運行時強制類型轉換而出現ClassCastException,類型轉化異常。

1.4 實例

JDK 1.5時增加了泛型,在很大的程度上方便在集合上的使用。

不使用泛型:

  1. public static void main(String[] args) { 
  2.  List list = new ArrayList(); 
  3.  list.add(11); 
  4.  list.add("ssss"); 
  5.  for (int i = 0; i < list.size(); i++) { 
  6.  System.out.println((String)list.get(i)); 
  7.  } 
  8.  } 

因為list類型是Object。所以int,String類型的數據都是可以放入的,也是都可以取出的。但是上述的代碼,運行的時候就會拋出類型轉化異常,這個相信大家都能明白。

使用泛型:

  1. public static void main(String[] args) { 
  2.  List<String> list = new ArrayList(); 
  3.  list.add("hahah"); 
  4.  list.add("ssss"); 
  5.  for (int i = 0; i < list.size(); i++) { 
  6.  System.out.println((String)list.get(i)); 
  7.  } 
  8.  } 

在上述的實例中,我們只能添加String類型的數據,否則編譯器會報錯。

2、泛型的使用

泛型的三種使用方式:泛型類,泛型方法,泛型接口

2.1 泛型類

泛型類概述:把泛型定義在類上

定義格式:

  1. public class 類名 <泛型類型1,...> { 
  2.   

注意事項:泛型類型必須是引用類型(非基本數據類型)

2.2 泛型方法

泛型方法概述:把泛型定義在方法上

定義格式:

public <泛型類型> 返回類型 方法名(泛型類型 變量名) { }

注意要點:

方法聲明中定義的形參只能在該方法里使用,而接口、類聲明中定義的類型形參則可以在整個接口、類中使用。當調用fun()方法時,根據傳入的實際對象,編譯器就會判斷出類型形參T所代表的實際類型。

  1. class Demo{  
  2.  public <T> T fun(T t){ // 可以接收任意類型的數據  
  3.  return t ; // 直接把參數返回  
  4.  }  
  5. };  
  6. public class GenericsDemo26{  
  7.  public static void main(String args[]){  
  8.  Demo d = new Demo() ; // 實例化Demo對象  
  9.  String str = d.fun("湯姆") ; // 傳遞字符串  
  10.  int i = d.fun(30) ; // 傳遞數字,自動裝箱  
  11.  System.out.println(str) ; // 輸出內容  
  12.  System.out.println(i) ; // 輸出內容  
  13.  }  
  14. }; 

2.3 泛型接口

泛型接口概述:把泛型定義在接口

定義格式:

  1. public interface 接口名<泛型類型> { 
  2.   

實例:

  1. /** 
  2.  * 泛型接口的定義格式: 修飾符 interface 接口名<數據類型> {} 
  3.  */ 
  4. public interface Inter<T> { 
  5.  public abstract void show(T t) ; 
  6. /** 
  7.  * 子類是泛型類 
  8.  */ 
  9. public class InterImpl<E> implements Inter<E> { 
  10.  @Override 
  11.  public void show(E t) { 
  12.  System.out.println(t); 
  13.  } 
  14. Inter<String> inter = new InterImpl<String>() ; 
  15. inter.show("hello") ; 

2.4 源碼中泛型的使用,下面是List接口和ArrayList類的代碼片段。

  1. //定義接口時指定了一個類型形參,該形參名為E 
  2. public interface List<E> extends Collection<E> { 
  3.  //在該接口里,E可以作為類型使用 
  4.  public E get(int index) {} 
  5.  public void add(E e) {}  
  6. //定義類時指定了一個類型形參,該形參名為E 
  7. public class ArrayList<E> extends AbstractList<E> implements List<E> { 
  8.  //在該類里,E可以作為類型使用 
  9.  public void set(E e) { 
  10.  ....................... 
  11.  } 

2.5 泛型類派生子類

父類派生子類的時候不能在包含類型形參,需要傳入具體的類型

錯誤的方式:

  1. public class A extends Container {} 

正確的方式:

  1. public class A extends Container {} 

也可以不指定具體的類型,系統就會把K,V形參當成Object類型處理

  1. public class A extends Container {} 

2.6 泛型構造器

構造器也是一種方法,所以也就產生了所謂的泛型構造器。

和使用普通方法一樣沒有區別,一種是顯示指定泛型參數,另一種是隱式推斷

  1. public class Person { 
  2.  public <T> Person(T t) { 
  3.  System.out.println(t); 
  4.  } 
  5.   

使用:

  1. public static void main(String[] args) { 
  2.  new Person(22);// 隱式 
  3.  new <String> Person("hello");//顯示 

特殊說明:

如果構造器是泛型構造器,同時該類也是一個泛型類的情況下應該如何使用泛型構造器:因為泛型構造器可以顯式指定自己的類型參數(需要用到菱形,放在構造器之前),而泛型類自己的類型實參也需要指定(菱形放在構造器之后),這就同時出現了兩個菱形了,這就會有一些小問題,具體用法再這里總結一下。 以下面這個例子為代表

  1. public class Person<E> { 
  2.  public <T> Person(T t) { 
  3.  System.out.println(t); 
  4.  } 

正確用法:

  1. public static void main(String[] args) { 
  2.  Person<String> person = new Person("sss"); 

PS:編譯器會提醒你怎么做的

2.7 高級通配符

2.7.1背景:

2.7.2 上界通配符

上界通配符顧名思義,表示的是類型的上界【包含自身】,因此通配的參數化類型可能是T或T的子類。

正因為無法確定具體的類型是什么,add方法受限(可以添加null,因為null表示任何類型),但可以從列表中獲取元素后賦值給父類型。如上圖中的第一個例子,第三個add()操作會受限,原因在于List和List是List的子類型。

它表示集合中的所有元素都是Animal類型或者其子類 List

這就是所謂的上限通配符,使用關鍵字extends來實現,實例化時,指定類型實參只能是extends后類型的子類或其本身。

例如:

這樣就確定集合中元素的類型,雖然不確定具體的類型,但最起碼知道其父類。然后進行其他操作。

  1. 它表示集合中的所有元素都是Animal類型或者其子類 
  2.  List<? extends Animal> 

2.7.3 下界通配符

下界通配符表示的是參數化類型是T的超類型(包含自身),層層至上,直至Object

編譯器無從判斷get()返回的對象的類型是什么,因此get()方法受限。但是可以進行add()方法,add()方法可以添加T類型和T類型的子類型,如第二個例子中首先添加了一個Cat類型對象,然后添加了兩個Cat子類類型的對象,這種方法是可行的,但是如果添加一個Animal類型的對象,顯然將繼承的關系弄反了,是不可行的。

它表示集合中的所有元素都是Cat類型或者其父類 List

這就是所謂的下限通配符,使用關鍵字super來實現,實例化時,指定類型實參只能是extends后類型的子類或其本身

例如

  1. //Animal是其父類 
  2. List<? super Cat> list = new ArrayList<Animal>(); 

2.7.4 無界通配符

任意類型,如果沒有明確,那么就是Object以及任意的Java類了

無界通配符用表示,?代表了任何的一種類型,能代表任何一種類型的只有null(Object本身也算是一種類型,但卻不能代表任何一種類型,所以List和List的含義是不同的,前者類型是Object,也就是繼承樹的最上層,而后者的類型完全是未知的)

3、泛型擦除

3.1 概念

編譯器編譯帶類型說明的集合時會去掉類型信息

3.2 驗證實例:

  1. public class GenericTest { 
  2.  public static void main(String[] args) { 
  3.  new GenericTest().testType(); 
  4.  } 
  5.  public void testType(){ 
  6.  ArrayList<Integer> collection1 = new ArrayList<Integer>(); 
  7.  ArrayList<String> collection2= new ArrayList<String>(); 
  8.   
  9.  System.out.println(collection1.getClass()==collection2.getClass()); 
  10.  //兩者class類型一樣,即字節碼一致 
  11.   
  12.  System.out.println(collection2.getClass().getName()); 
  13.  //class均為java.util.ArrayList,并無實際類型參數信息 
  14.  } 

輸出結果:

  1. true 
  2. java.util.ArrayList 

分析:

這是因為不管為泛型的類型形參傳入哪一種類型實參,對于Java來說,它們依然被當成同一類處理,在內存中也只占用一塊內存空間。從Java泛型這一概念提出的目的來看,其只是作用于代碼編譯階段,在編譯過程中,對于正確檢驗泛型結果后,會將泛型的相關信息擦出,也就是說,成功編譯過后的class文件中是不包含任何泛型信息的。泛型信息不會進入到運行時階段。

在靜態方法、靜態初始化塊或者靜態變量的聲明和初始化中不允許使用類型形參。由于系統中并不會真正生成泛型類,所以instanceof運算符后不能使用泛型類

4、泛型與反射

把泛型變量當成方法的參數,利用Method類的getGenericParameterTypes方法來獲取泛型的實際類型參數

例子:

  1. public class GenericTest { 
  2.  public static void main(String[] args) throws Exception { 
  3.  getParamType(); 
  4.  } 
  5.   
  6.  /*利用反射獲取方法參數的實際參數類型*/ 
  7.  public static void getParamType() throws NoSuchMethodException{ 
  8.  Method method = GenericTest.class.getMethod("applyMap",Map.class); 
  9.  //獲取方法的泛型參數的類型 
  10.  Type[] types = method.getGenericParameterTypes(); 
  11.  System.out.println(types[0]); 
  12.  //參數化的類型 
  13.  ParameterizedType pType = (ParameterizedType)types[0]; 
  14.  //原始類型 
  15.  System.out.println(pType.getRawType()); 
  16.  //實際類型參數 
  17.  System.out.println(pType.getActualTypeArguments()[0]); 
  18.  System.out.println(pType.getActualTypeArguments()[1]); 
  19.  } 
  20.  /*供測試參數類型的方法*/ 
  21.  public static void applyMap(Map<Integer,String> map){ 
  22.  } 

輸出結果:

  1. java.util.Map<java.lang.Integer, java.lang.String> 
  2. interface java.util.Map 
  3. class java.lang.Integer 
  4. class java.lang.String 

通過反射繞開編譯器對泛型的類型限制

  1. public static void main(String[] args) throws Exception { 
  2.         //定義一個包含int的鏈表 
  3.         ArrayList<Integer> al = new ArrayList<Integer>(); 
  4.         al.add(1); 
  5.         al.add(2); 
  6.         //獲取鏈表的add方法,注意這里是Object.class,如果寫int.class會拋出NoSuchMethodException異常 
  7.         Method m = al.getClass().getMethod("add", Object.class); 
  8.         //調用反射中的add方法加入一個string類型的元素,因為add方法的實際參數是Object 
  9.         m.invoke(al, "hello"); 
  10.         System.out.println(al.get(2)); 
  11.     } 

5 泛型的限制

5.1 模糊性錯誤

對于泛型類User

  1. public class User<K, V> { 
  2.   
  3.  public void show(K k) { // 報錯信息:'show(K)' clashes with 'show(V)'; both methods have same erasure 
  4.   
  5.  } 
  6.  public void show(V t) { 
  7.  } 

由于泛型擦除,二者本質上都是Obejct類型。方法是一樣的,所以編譯器會報錯。

換一個方式:

  1. public class User<K, V> { 
  2.  public void show(String k) { 
  3.  } 
  4.  public void show(V t) { 
  5.  } 

使用結果:

 

Java 泛型全解 - 絕對最詳細

 

可以正常的使用5.2 不能實例化類型參數

編譯器也不知道該創建那種類型的對象

  1. public class User<K, V> { 
  2.  private K key = new K(); // 報錯:Type parameter 'K' cannot be instantiated directly 

5.3 對靜態成員的限制

靜態方法無法訪問類上定義的泛型;如果靜態方法操作的類型不確定,必須要將泛型定義在方法上。

如果靜態方法要使用泛型的話,必須將靜態方法定義成泛型方法。

  1. public class User<T> { 
  2.  //錯誤 
  3.  private static T t; 
  4.  //錯誤 
  5.  public static T getT() { 
  6.  return t; 
  7.  } 
  8.  //正確 
  9.  public static <K> void test(K k) { 
  10.  } 

5.4 對泛型數組的限制

不能實例化元素類型為類型參數的數組,但是可以將數組指向類型兼容的數組的引用

  1. public class User<T> { 
  2.  private T[] values
  3.  public User(T[] values) { 
  4.  //錯誤,不能實例化元素類型為類型參數的數組 
  5.  this.values = new T[5]; 
  6.  //正確,可以將values 指向類型兼容的數組的引用 
  7.  this.values = values
  8.  } 

5.5 對泛型異常的限制

泛型類不能擴展 Throwable,意味著不能創建泛型異常類

責任編輯:武曉燕 來源: 今日頭條
相關推薦

2022-08-08 08:25:21

Javajar 文件

2023-09-22 17:36:37

2021-01-28 22:31:33

分組密碼算法

2020-05-22 08:16:07

PONGPONXG-PON

2018-06-07 13:17:12

契約測試單元測試API測試

2023-09-29 08:58:38

2020-12-11 11:11:44

原子類JavaCAS

2021-07-08 11:22:55

Java異常處理

2019-02-13 14:15:59

Linux版本Fedora

2021-08-04 09:32:05

Typescript 技巧Partial

2021-01-29 08:32:21

數據結構數組

2021-02-06 08:34:49

函數memoize文檔

2018-11-29 09:13:47

CPU中斷控制器

2022-11-01 08:46:20

責任鏈模式對象

2023-07-06 13:56:14

微軟Skype

2023-05-15 08:38:58

模板方法模式

2020-10-15 06:56:51

MySQL排序

2020-08-12 08:34:16

開發安全We

2021-01-01 09:01:05

前端組件化設計

2022-11-26 00:00:06

裝飾者模式Component
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 精品欧美一区免费观看α√ | 国产成人免费视频网站高清观看视频 | 国产婷婷综合 | 国产精品国产精品 | 麻豆精品久久 | 毛片一级片| 99精品久久久 | 天天插天天射天天干 | 国产精品久久久亚洲 | 国外激情av | 二区中文 | www.干| 欧美国产日韩在线观看成人 | 九九九精品视频 | 日韩激情视频一区 | 久久之精品 | 日韩免费视频 | 国产免费高清 | 精品欧美一区二区三区久久久 | 国产精品久久九九 | 久久久精品综合 | 欧美激情精品久久久久久变态 | 成人欧美一区二区 | 亚洲精品国产a久久久久久 午夜影院网站 | 精彩视频一区二区三区 | 国产精品久久久99 | 中文字幕一区二区三区四区 | 欧美一级在线视频 | 色综合区| 免费黄色成人 | www.成人.com| 日韩中文字幕在线观看视频 | 天天综合国产 | 亚洲欧美激情精品一区二区 | 亚洲一区二区视频在线播放 | 欧美一级欧美三级在线观看 | 韩日精品在线观看 | 久久精彩视频 | 一区二区视频 | 韩国理论电影在线 | 久久久久香蕉视频 |