偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java泛型你必須知道的知識(shí)

開發(fā) 后端
Java 泛型(generics)是 JDK 5 中引入的一個(gè)新特性, 泛型提供了編譯時(shí)類型安全檢測(cè)機(jī)制,該機(jī)制允許程序員在編譯時(shí)檢測(cè)到非法的類型。

[[384110]]

 一 什么是泛型

Java 泛型(generics)是 JDK 5 中引入的一個(gè)新特性, 泛型提供了編譯時(shí)類型安全檢測(cè)機(jī)制,該機(jī)制允許程序員在編譯時(shí)檢測(cè)到非法的類型。

簡(jiǎn)單理解就是:泛型指定編譯時(shí)的類型,減少運(yùn)行時(shí)由于對(duì)象類型不匹配引發(fā)的異常。其主要用途是提高我們的代碼的復(fù)用率。

我們Java標(biāo)準(zhǔn)庫中的ArrayList就是泛型使用的典型應(yīng)用:

  1. public class ArrayList<E> extends AbstractList<E> implements List<E>, RandomAccess, Cloneable, java.io.Serializable { 
  2.         
  3.       ...... 
  4.  
  5.     public ArrayList(Collection<? extends E> c) { 
  6.         elementData = c.toArray(); 
  7.         if ((size = elementData.length) != 0) { 
  8.             // c.toArray might (incorrectly) not return Object[] (see 6260652) 
  9.             if (elementData.getClass() != Object[].class) 
  10.                 elementData = Arrays.copyOf(elementData, size, Object[].class); 
  11.         } else { 
  12.             // replace with empty array. 
  13.             this.elementData = EMPTY_ELEMENTDATA; 
  14.         } 
  15.     } 
  16.  
  17.     public void sort(Comparator<? super E> c) { 
  18.         final int expectedModCount = modCount; 
  19.         Arrays.sort((E[]) elementData, 0, size, c); 
  20.         if (modCount != expectedModCount) { 
  21.             throw new ConcurrentModificationException(); 
  22.         } 
  23.         modCount++; 
  24.     } 
  25.     
  26.   ..... 
  27.  
  28.     public E get(int index) { 
  29.         rangeCheck(index); 
  30.  
  31.         return elementData(index); 
  32.     } 
  33.  
  34.     public boolean add(E e) { 
  35.         ensureCapacityInternal(size + 1);  // Increments modCount!! 
  36.         elementData[size++] = e; 
  37.         return true
  38.     } 
  39.  
  •  源碼中,ArrayList中的E稱為類型參數(shù)變量,而整個(gè)ArrayList我們稱為泛型類型。 我們可以指定除基本類型之外的任何類型,如:ArrayList
  • 源碼中Collection 中? 通配符類型 表示類型的上界,表示參數(shù)化類型的可能是T 或是 T的子類。
  • 源碼中Comparator 表示類型下界(Java Core中叫超類型限定),表示參數(shù)化類型是此類型的超類型(父類型),直至Object。

二 extends和super通配符

在定義泛型類型Generic的時(shí)候,也可以使用extends通配符來限定T的類型:

  1. public class Generic<T extends Number> { ... } 

現(xiàn)在,我們只能定義:

  1. Generic<Number> p1 = null
  2. Generic<Integer> p2 = new Generic<>(1, 2); 
  3. Generic<Double> p3 = null

因?yàn)镹umber、Integer和Double都符合。

非Number類型將無法通過編譯:

  1. Generic<String> p1 = null; // compile error! 
  2. Generic<Object> p2 = null; // compile error! 

 因?yàn)镾tring、Object都不符合,因?yàn)樗鼈儾皇荖umber類型或Number的子類。

我們看一個(gè)例子:

  1. public class Test { 
  2.  
  3.     static class Food { 
  4.  
  5.     } 
  6.  
  7.     static class Fruit extends Food { 
  8.     } 
  9.  
  10.     static class Apple extends Fruit { 
  11.     } 
  12.  
  13.     static class Orange extends Fruit { 
  14.     } 
  15.  
  16.     public void testExtend() { 
  17.         List<? extends Fruit> list = new ArrayList<Apple>(); 
  18.  
  19.         //無法安全添加任何具有實(shí)際意義的元素,報(bào)錯(cuò),extends為上界通配符,只能取值,不能放. 
  20.         //因?yàn)镕ruit的子類不只有Apple還有Orange,這里不能確定具體的泛型到底是Apple還是Orange,所以放入任何一種類型都會(huì)報(bào)錯(cuò) 
  21.  
  22.         //list.add(new Apple()); 
  23.         //list.add(new Orange()); 
  24.  
  25.         //可以添加null,因?yàn)?/span>null可以表示任何類型 
  26.         list.add(null); 
  27.  
  28.         //可以正常獲取,用java多態(tài) 
  29.         Food foot = list.get(0); 
  30.         Apple apple = (Apple) list.get(0); 
  31.     } 
  32.  
  33.     public void testSuper() { 
  34.         List<? super Fruit> list = new ArrayList<Fruit>(); 
  35.  
  36.         //super為下界通配符,可以存放元素,但是也只能存放當(dāng)前類或者子類的實(shí)例,以當(dāng)前的例子來講, 
  37.         list.add(new Fruit()); 
  38.         list.add(new Apple()); 
  39.  
  40.         //無法確定Fruit的父類是否只有Food一個(gè)(Object是超級(jí)父類) 
  41.         //因此放入Food的實(shí)例編譯不通過,只能放自己的實(shí)例 或者根據(jù)java多態(tài)的特性放子類實(shí)例 
  42.         //list.add(new Food()); 
  43.         //List<? super Fruit> list2 = new ArrayList<Apple>(); 
  44.         //Fruit fruit = list.get(0); //不能確定返回類型 
  45.  
  46.     } 
  47.  

 在testExtend方法中,因?yàn)榉盒椭杏玫氖莈xtends,在向list中存放元素的時(shí)候,我們并不能確定List中的元素的具體類型,即可能是Apple也可能是Orange。因此調(diào)用add方法時(shí),不論傳入new Apple()還是new Orange(),都會(huì)出現(xiàn)編譯錯(cuò)誤。

理解了extends之后,再看super就很容易理解了,即我們不能確定testSuper方法的參數(shù)中的泛型是Fruit的哪個(gè)父類,因此在調(diào)用get方法時(shí)只能返回Object類型。結(jié)合extends可見,在獲取泛型元素時(shí),使用extends獲取到的是泛型中的上邊界的類型(本例子中為Fruit),范圍更小。

總結(jié):

  • 在使用泛型時(shí),存取元素時(shí)用super。
  • 獲取元素時(shí),用extends。

有了上面的結(jié)論我們看下Java標(biāo)準(zhǔn)庫的Collections類定義的copy()方法,這個(gè)copy()方法的定義就完美地展示了extends和super的意圖:

  • copy()方法內(nèi)部不會(huì)讀取dest,因?yàn)椴荒苷{(diào)用dest.get()來獲取T的引用;
  • copy()方法內(nèi)部也不會(huì)修改src,因?yàn)椴荒苷{(diào)用src.add(T)。
  1. public class Collections { 
  2.     // 把src的每個(gè)元素復(fù)制到dest中: 
  3.     public static <T> void copy(List<? super T> dest, List<? extends T> src) { 
  4.         for (int i=0; i<src.size(); i++) { 
  5.             T t = src.get(i); 
  6.             dest.add(t); 
  7.         } 
  8.     } 

 三 泛型擦除

Java的泛型是偽泛型,這是因?yàn)镴ava在編譯期間,所有的泛型信息都會(huì)被擦掉,正確理解泛型概念的首要前提是理解類型擦除。Java的泛型基本上都是在編譯器這個(gè)層次上實(shí)現(xiàn)的,在生成的字節(jié)碼中是不包含泛型中的類型信息的,使用泛型的時(shí)候加上類型參數(shù),在編譯器編譯的時(shí)候會(huì)去掉,這個(gè)過程成為類型擦除

我們看一個(gè)示例:

  1. public class Test2 { 
  2.  
  3.     public static void main(String[] args) { 
  4.         Map<String, Animal> map = new HashMap<>(); 
  5.         Animal animal = new Animal(); 
  6.         animal.setVegetarian(true); 
  7.         animal.setEats("fish"); 
  8.         map.put("cat", animal); 
  9.  
  10.         String json = new Gson().toJson(map); 
  11.         System.out.println(json); 
  12.  
  13.         Map<String, Animal> jsonToMap = fromJson(json); 
  14.         System.out.println(jsonToMap); 
  15.  
  16.         Animal animal1 = jsonToMap.get("cat"); 
  17.         System.out.println(animal1.getEats()); 
  18.     } 
  19.  
  20.     public static <T> T fromJson(String str) { 
  21.         return new Gson().fromJson(str, new TypeToken<T>() { 
  22.         }.getType()); 
  23.     } 
  24.  

 上的代碼運(yùn)行會(huì)提示如下異常:

  1. Exception in thread "main" java.lang.ClassCastException: com.google.gson.internal.LinkedTreeMap cannot be cast to com.uaf.rabbitmq.producer.Animal 
  2.     at com.uaf.rabbitmq.producer.Test2.main(Test2.java:30) 

 異常原因主要是這句:new Gson().fromJson(str, new TypeToken() {}.getType());

這句在實(shí)際執(zhí)行的時(shí)候,List中的T并未傳入實(shí)際的泛型參數(shù),導(dǎo)致Gson按照LinkedTreeMap來解析JSON,以致發(fā)生了錯(cuò)誤;這就是一個(gè)在編譯期泛型類型擦除所導(dǎo)致的問題;

解決這個(gè)問題我們需要修改fromJson方法

  1. public class Test2 { 
  2.  
  3.     public static void main(String[] args) { 
  4.         Map<String, Animal> map = new HashMap<>(); 
  5.         Animal animal = new Animal(); 
  6.         animal.setVegetarian(true); 
  7.         animal.setEats("fish"); 
  8.         map.put("cat", animal); 
  9.  
  10.         String json = new Gson().toJson(map); 
  11.         System.out.println(json); 
  12.  
  13.         Map<String, Animal> jsonToMap = fromJson(json,  
  14.         new TypeToken<Map<String, Animal>>() {}.getType()); 
  15.         System.out.println(jsonToMap); 
  16.  
  17.         Animal animal1 = jsonToMap.get("cat"); 
  18.         System.out.println(animal1.getEats()); 
  19.  
  20.     } 
  21.  
  22.     public static <T> T fromJson(String str, Type type) { 
  23.         return new Gson().fromJson(str, type); 
  24.     } 
  25.  

 在Gson中提供了TypeToken解決泛型運(yùn)行時(shí)類型擦除問題,TypeToken 這個(gè)類來幫助我們捕獲像Map這樣的泛型信息。上文創(chuàng)建了一個(gè)匿名內(nèi)部類,這樣Java編譯器就會(huì)把泛型信息編譯到這個(gè)匿名內(nèi)部類里,然后在運(yùn)行時(shí)就可以被getType()方法用反射API提取到。

 

責(zé)任編輯:姜華 來源: 今日頭條
相關(guān)推薦

2017-12-07 15:47:25

2017-12-07 15:28:36

2017-10-11 15:50:18

光纖通信傳輸

2011-05-13 11:41:55

2020-02-28 14:05:00

Linuxshell命令

2021-06-29 15:56:39

MYSQL開發(fā)數(shù)據(jù)庫

2012-09-29 10:29:56

.Net內(nèi)存分配繼承

2012-09-29 09:22:24

.NETGC內(nèi)存分配

2021-10-29 08:44:22

推拉機(jī)制面試broker

2011-11-30 09:09:13

王濤Windows Pho移動(dòng)開發(fā)

2015-06-29 09:40:10

Rails新特性

2024-03-29 13:17:03

Docker數(shù)據(jù)卷Volume

2015-07-23 10:37:13

Linux命令

2021-10-17 13:10:56

函數(shù)TypeScript泛型

2012-11-05 09:19:37

2011-05-11 15:28:05

2019-05-30 08:25:50

5G4G網(wǎng)絡(luò)

2011-12-16 17:05:58

2019-01-08 10:29:12

BeautifulSoPython第三庫

2012-02-08 09:44:05

ChromeAndroid
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)