精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Java的“泛型”特性,你以為自己會了?

開發(fā) 后端
使用Java的小伙伴,對于Java的一些高級特性一定再熟悉不過了,例如集合、反射、泛型、注解等等,這些可以說我們在平時開發(fā)中是經(jīng)常使用到的,尤其是集合,基本是只要寫代碼沒有用不到的,今天我們先來談?wù)劮盒汀?/div>

[[381366]]

使用Java的小伙伴,對于Java的一些高級特性一定再熟悉不過了,例如集合、反射、泛型、注解等等,這些可以說我們在平時開發(fā)中是經(jīng)常使用到的,尤其是集合,基本是只要寫代碼沒有用不到的,今天我們先來談?wù)劮盒汀?/p>

1. 定義

在了解一個事物之前,我們必定要先知道他的定義,所以我們就從定義開始,去一步一步揭開泛型的神秘面紗。

# 泛型(generics)

他是 JDK5 中引入的一個新特性,泛型提供了編譯時類型安全監(jiān)測機制,該機制允許我們在編譯時檢測到非法的類型數(shù)據(jù)結(jié)構(gòu)。泛型的本質(zhì)就是參數(shù)化類型,也就是所操作的數(shù)據(jù)類型被指定為一個參數(shù)# 常見的泛型的類型表示上面的 T 僅僅類似一個形參的作用,名字實際上是可以任意起的,但是我們寫代碼總該是要講究可讀性的。常見的參數(shù)通常有 :E - Element (在集合中使用,因為集合中存放的是元素)T - Type(表示Java 類,包括基本的類和我們自定義的類)K - Key(表示鍵,比如Map中的key)V - Value(表示值)? - (表示不確定的java類型)但是泛型的參數(shù)只能是類類型,不能是基本的數(shù)據(jù)類型,他的類型一定是自O(shè)bject的

注意:泛型不接受基本數(shù)據(jù)類型,換句話說,只有引用類型才能作為泛型方法的實際參數(shù)

2. 為什么要使用泛型?

說到為什么要使用,那肯定是找一大堆能說服自己的優(yōu)點啊。

# 泛型的引入,是java語言的來講是一個較大的功能增強。同時對于編譯器也帶來了一定的增強,為了支持泛型,java的類庫都做相應(yīng)的修改以支持泛型的特性。(科普:實際上java泛型并不是 jdk5(2004發(fā)布了jdk5) 才提出來的,早在1999年的時候,泛型機制就是java最早的規(guī)范之一)

另外,泛型還具有以下的優(yōu)點:

# 1.提交了java的類型安全

泛型在很大程度上來提高了java的程序安全。例如在沒有泛型的情況下,很容易將字符串 123 轉(zhuǎn)成 Integer 類型的 123 亦或者 Integer 轉(zhuǎn)成 String,而這樣的錯誤是在編譯期無法檢測。而使用泛型,則能很好的避免這樣的情況發(fā)生。

# 2.不需要煩人的強制類型轉(zhuǎn)換

泛型之所以能夠消除強制類型轉(zhuǎn)換,那是因為程序員在開發(fā)的時候就已經(jīng)明確了自己使用的具體類型,這不但提高了代碼的可讀性,同樣增加了代碼的健壯性。

# 提高了代碼的重用性

泛型的程序設(shè)計,意味著編寫的代碼可以被很多不同類型的對象所重用

在泛型規(guī)范正式發(fā)布之前,泛型的程序設(shè)計是通過繼承來實現(xiàn)的,但是這樣子有兩個嚴(yán)重的問題:

① 取值的時候需要強制類型轉(zhuǎn)換,否則拿到的都是 Object

② 編譯期不會有錯誤檢查

我們來看下這兩個錯誤的產(chǎn)生

2.1 編譯期不會有錯誤檢查

  1. public class DonCheckInCompile { 
  2.    public static void main(String[] args) { 
  3.        List list = new ArrayList(); 
  4.        list.add("a"); 
  5.        list.add(3); 
  6.        System.out.println(list); 
  7.   } 

程序不但不會報錯,還能正常輸出

 

2.2 強制類型轉(zhuǎn)換

  1. public class DonCheckInCompile { 
  2.    public static void main(String[] args) { 
  3.        List list = new ArrayList(); 
  4.        list.add("a"); 
  5.        list.add(3); 
  6.        for (Object o : list) { 
  7.            System.out.println((String)o); 
  8.       } 
  9.   } 

 

因為你并不知道實際集合中的元素到底是哪些類型的,所以在使用的時候也是不確定的,如果在強轉(zhuǎn)的時候,那必然會帶來意想不到的錯誤,這樣潛在的問題就好像是定時炸彈,肯定是不允許發(fā)生的。所以這就更體現(xiàn)了泛型的重要性。

3. 泛型方法

在 java 中,泛型方法可以使用在成員方法、構(gòu)造方法和靜態(tài)方法中。語法如下:

public <申明泛型的類型> 類型參數(shù) fun();如 public T fun(T t);這里的 T 表示一個泛型類型,而 表示我們定義了一個類型為 T 的類型,這樣的 T 類型就可以直接使用了,且 需要放在方法的返回值類型之前。T 即在申明的時候是不知道具體的類型的,只有的使用的時候才能明確其類型,T 不是一個類,但是可以當(dāng)作是一種類型來使用。

下面來通過具體的例子來解釋說明,以下代碼將數(shù)組中的指定的兩個下標(biāo)位置的元素進(jìn)行交換(不要去關(guān)注實際的需求是什么),第一種 Integer 類型的數(shù)組

  1. public class WildcardCharacter { 
  2.    public static void main(String[] args) { 
  3.        Integer[] arrInt = {1, 2, 3, 4, 5, 6, 7, 8, 9}; 
  4.        change(arrInt, 0, 8); 
  5.        System.out.println("arr = " + Arrays.asList(arrInt)); 
  6.   } 
  7.  
  8.    /** 
  9.     * 將數(shù)組中的指定兩個下標(biāo)位置的元素交換 
  10.     * 
  11.     * @param arr         數(shù)組 
  12.     * @param firstIndex 第一個下標(biāo) 
  13.     * @param secondIndex 第二個下標(biāo) 
  14.     */ 
  15.    private static void change(Integer[] arr, int firstIndex, int secondIndex) { 
  16.        int tmp = arr[firstIndex]; 
  17.        arr[firstIndex] = arr[secondIndex]; 
  18.        arr[secondIndex] = tmp; 
  19.   } 

 

第二種是 String 類型的數(shù)組

 

編譯直接都不會通過,那是必然的,因為方法定義的參數(shù)就是 Integer[] 結(jié)果你傳一個String[],玩呢。。。所以這個時候只能是再定義一個參數(shù)類型是 String[]的。

那要是再來一個 Double 呢?Boolean 呢?是不是這就產(chǎn)生問題了,雖然說這種問題不是致命的,多寫一些重復(fù)的代碼就能解決,但這勢必導(dǎo)致代碼的冗余和維護(hù)成本的增加。所以這個時候泛型的作用就體現(xiàn)了,我們將其改成泛型的方式。

  1. /** 
  2.    * @param t           參數(shù)類型 T 
  3.    * @param firstIndex 第一個下標(biāo) 
  4.    * @param secondIndex 第二個下標(biāo) 
  5.    * @param <T>         表示定義了一個類型 為 T 的類型,否則沒人知道 T 是什么,編譯期也不知道 
  6.    */ 
  7.   private static <T> void changeT(T[] t, int firstIndex, int secondIndex) { 
  8.       T tmp = t[firstIndex]; 
  9.       t[firstIndex] = t[secondIndex]; 
  10.       t[secondIndex] = tmp; 
  11.  } 

接下來調(diào)用就簡單了

  1. public static void main(String[] args) { 
  2.     //首先定義一個Integer類型的數(shù)組 
  3.        Integer[] arrInt = {1, 2, 3, 4, 5, 6, 7, 8, 9}; 
  4.     //將第 1 個和第 9 個位置的元素進(jìn)行交換 
  5.        changeT(arrInt, 0, 8); 
  6.        System.out.println("arrInt = " + Arrays.asList(arrInt)); 
  7.        // 然后在定義一個String類型的數(shù)組 
  8.        String[] arrStr = {"a""b""c""d""e""f""g"}; 
  9.   //將第 1 個和第 2 個位置的元素進(jìn)行交換 
  10.        changeT(arrStr, 0, 1); 
  11.         System.out.println("arrStr = " + Arrays.asList(arrStr)); 
  12.   } 

 

問題迎刃而解,至于普通的泛型方法和靜態(tài)的泛型方法是一樣的使用,只不過是一個數(shù)據(jù)類一個屬于類的實例的,在使用上區(qū)別不大(但是需要注意的是如果在泛型類中 靜態(tài)泛型方法是不能使用類泛型中的泛型類型的,這個在下文的泛型類中會詳細(xì)介紹的)。

最后在來看下構(gòu)造方法

  1. public class Father { 
  2.     public <T> Father(T t) { 
  3.     } 

然后假設(shè)他有一個子類是這樣子的

  1. class Son extends Father { 
  2.  
  3.     public <T> Son(T t) { 
  4.         super(t); 
  5.     } 

這里強調(diào)一下,因為在 Father 類中是沒有無參構(gòu)造器的,取而代之的是一個有參的構(gòu)造器,只不過這個構(gòu)造方法是一個泛型的方法,那這樣子的子類必然需要顯示的指明構(gòu)造器了。

  • 通過泛型方法獲取集合中的元素測試

既然說泛型是在申明的時候類型不是重點,只要事情用的時候確定就可以下,那你看下面這個怎么解釋?

 

此時想往集合中添加元素,卻提示這樣的錯誤,連編譯都過不了。這是為什么?

因為此時集合 List 的 add 方法,添加的類型為 T,但是很顯然 T 是一個泛型,真正的類型是在使用時候才能確定的,但是 在 add 的并不能確定 T 的類型,所以根本就無法使用 add 方法,除非 list.add(null),但是這卻沒有任何意義。

4. 泛型類

先來看一段這樣的代碼,里面的使用到了多個泛型的方法,無需關(guān)注方法到底做了什么

  1. public class GenericClassTest{ 
  2.     public static void main(String[] args) { 
  3.         //首先定義一個Integer類型的數(shù)組 
  4.         Integer[] arrInt = {1, 2, 3, 4, 5, 6, 7, 8, 9}; 
  5.         //將第 1 個和第 9 個位置的元素進(jìn)行交換 
  6.  
  7.        new GenericClassTest().changeT(arrInt, 0, 8); 
  8.         System.out.println("arrInt = " + Arrays.asList(arrInt)); 
  9.  
  10.         List<String> list = Arrays.asList("a""b"); 
  11.         testIter(list); 
  12.     } 
  13.  
  14.     /** 
  15.      * @param t           參數(shù)類型 T 
  16.      * @param firstIndex  第一個下標(biāo) 
  17.      * @param secondIndex 第二個下標(biāo) 
  18.      * @param <T>         表示定義了一個類型 為 T 的類型,否則沒人知道 T 是什么,編譯期也不知道 
  19.      */ 
  20.     private <T> void changeT(T[] t, int firstIndex, int secondIndex) { 
  21.         T tmp = t[firstIndex]; 
  22.         t[firstIndex] = t[secondIndex]; 
  23.         t[secondIndex] = tmp; 
  24.     } 
  25.  
  26.     /** 
  27.      * 遍歷集合 
  28.      * 
  29.      * @param list 集合 
  30.      * @param <T>  表示定義了一個類型 為 T 的類型,否則沒人知道 T 是什么,編譯期也不知道 
  31.      */ 
  32.     private static <T> void testIter(List<T> list) { 
  33.         for (T t : list) { 
  34.             System.out.println("t = " + t); 
  35.         } 
  36.     } 

可以看到里面的 是不是每個方法都需要去申明一次,那要是 100 個方法呢?那是不是要申明 100 次的,這樣時候泛型類也就應(yīng)用而生了。那泛型類的形式是什么樣子的呢?請看代碼

  1. public class GenericClazz<T>{ 
  2.     //這就是一個最基本的泛型類的樣子 

下面我們將剛剛的代碼優(yōu)化如下,但是這里不得不說一個很基礎(chǔ),但是卻很少有人注意到的問題,請看下面的截圖中的文字描述部分。

# 為什么實例方法可以,而靜態(tài)方法卻報錯?1. 首先告訴你結(jié)論:靜態(tài)方法不能使用類定義的泛型,而是應(yīng)該單獨定義泛型2. 到這里估計很多小伙伴就瞬間明白了,因為靜態(tài)方法是通過類直接調(diào)用的,而普通方法必須通過實例來調(diào)用,類在調(diào)用靜態(tài)方法的時候,后面的泛型類還沒有被創(chuàng)建,所以肯定不能這么去調(diào)用的

 

所以說這個泛型類中的靜態(tài)方法直接這么寫就可以啦

  1. /** 
  2.      * 遍歷集合 
  3.      * 
  4.      * @param list 集合 
  5.      */ 
  6.     private static <K> void testIter(List<K> list) { 
  7.         for (K t : list) { 
  8.             System.out.println("t = " + t); 
  9.         } 
  10.     } 
  • 多個泛型類型同時使用

我們知道 Map 是鍵值對形式存在,所以如果對 Map 的 Key 和 Value 都使用泛型類型該怎么辦?一樣的使用,一個靜態(tài)方法就可以搞定了,請看下面的代碼

  1. public class GenericMap { 
  2.  
  3.     private static <K, V> void mapIter(Map<K, V> map) { 
  4.         for (Map.Entry<K, V> kvEntry : map.entrySet()) { 
  5.             K key = kvEntry.getKey(); 
  6.             V value = kvEntry.getValue(); 
  7.             System.out.println(key + ":" + value); 
  8.         } 
  9.     } 
  10.  
  11.     public static void main(String[] args) { 
  12.         Map<String, String> mapStr = new HashMap<>(); 
  13.         mapStr.put("a""aa"); 
  14.         mapStr.put("b""bb"); 
  15.         mapStr.put("c""cc"); 
  16.         mapIter(mapStr); 
  17.         System.out.println("======"); 
  18.         Map<Integer, String> mapInteger = new HashMap<>(); 
  19.         mapInteger.put(1, "11"); 
  20.         mapInteger.put(2, "22"); 
  21.         mapInteger.put(3, "33"); 
  22.         mapIter(mapInteger); 
  23.     } 

 

到此,泛型的常規(guī)的方法和泛型類已經(jīng)介紹為了。

5. 通配符

通配符 ? 即占位符的意思,也就是在使用期間是無法確定其類型的,只要在將來實際使用的時再指明類型,它有三種形式

  • <?> 無限定的通配符。是讓泛型能夠接受未知類型的數(shù)據(jù)
  • < ? extends E>有上限的通配符。能接受指定類及其子類類型的數(shù)據(jù),E就是該泛型的上邊界
  • <? super E>有下限的通配符。能接受指定類及其父類類型的數(shù)據(jù),E就是該泛型的下邊界

5.1 通配符之

上面剛剛說到了使用一個類型來表示反省類型是必須要申明的,也即 ,那是不是不申明就不能使用泛型呢?當(dāng)然不是,這小節(jié)介紹的 就是為了解決這個問題的。

表示,但是話又說話來了,那既然可以不去指明具體類型,那 ? 就不能表示一個具體的類型也就是說如果按照原來的方式這么去寫,請看代碼中的注釋

 

而又因為任何類型都是 Object 的子類,所以,這里可以使用 Object 來接收,對于 ?的具體使用會在下面兩小節(jié)介紹

 

另外,大家要搞明白泛型和通配符不是一回事

5.2 通配符之 <? extend E>

<? extend E> 表示有上限的通配符,能接受其類型和其子類的類型 E 指上邊界,還是寫個例子來說明

  1. public class GenericExtend { 
  2.     public static void main(String[] args) { 
  3.         List<Father> listF = new ArrayList<>(); 
  4.         List<Son> listS = new ArrayList<>(); 
  5.         List<Daughter> listD = new ArrayList<>(); 
  6.         testExtend(listF); 
  7.         testExtend(listS); 
  8.         testExtend(listD); 
  9.     } 
  10.  
  11.     private static <T> void testExtend(List<? extends Father> list) {} 
  12.  
  13. class Father {} 
  14.  
  15. class Daughter extends Father{} 
  16.  
  17. class Son extends Father {     

這個時候一切都還是很和平的,因為大家都遵守著預(yù)定,反正 List 中的泛型要么是 Father 類,要么是 Father 的子類。但是這個時候如果這樣子來寫(具體原因已經(jīng)在截圖中寫明了)

 

5.3 通配符之 <?super E>

表示有下限的通配符。也就說能接受指定類型及其父類類型,E 即泛型類型的下邊界,直接上來代碼然后來解釋

  1. public class GenericSuper { 
  2.  
  3.     public static void main(String[] args) { 
  4.         List<Son> listS = new Stack<>(); 
  5.         List<Father> listF = new Stack<>(); 
  6.         List<GrandFather> listG = new Stack<>(); 
  7.         testSuper(listS); 
  8.         testSuper(listF); 
  9.         testSuper(listG); 
  10.          
  11.     } 
  12.     private static void testSuper(List<? super Son> list){} 
  13. class Son extends Father{} 
  14. class Father extends GrandFather{} 
  15. class GrandFather{} 

因為 List list 接受的類型只能是 Son 或者是 Son 的父類,而 Father 和 GrandFather 又都是 Son 的父類,所以以上程序是沒有任何問題的,但是如果再來一個類是 Son 的子類(如果不是和 Son 有關(guān)聯(lián)的類那更不行了),那結(jié)果會怎么樣?看下圖,相關(guān)重點已經(jīng)在圖中詳細(xì)說明

 

好了,其實泛型說到這里基本就差不多了,我們平時開發(fā)能遇到的問題和不常遇見的問題本文都基本講解到了。最后我們再來一起看看泛型的另一個特性:泛型擦除。

6. 泛型擦除

先來看下泛型擦除的定義

# 泛型擦除 因為泛型的信息只存在于 java 的編譯階段,編譯期編譯完帶有 java 泛型的程序后,其生成的 class 文件中與泛型相關(guān)的信息會被擦除掉,以此來保證程序運行的效率并不會受影響,也就說泛型類型在 jvm 中和普通類是一樣的。

別急,知道你看完概念肯定還是不明白什么叫泛型擦除,舉個例子

  1. public class GenericWipe { 
  2.     public static void main(String[] args) { 
  3.         List<String> listStr = new ArrayList<>(); 
  4.         List<Integer> listInt = new ArrayList<>(); 
  5.         List<Double> listDou = new ArrayList<>(); 
  6.  
  7.         System.out.println(listStr.getClass()); 
  8.         System.out.println(listInt.getClass()); 
  9.         System.out.println(listDou.getClass()); 
  10.  
  11.     } 

 

這也就是說 java 泛型在生成字節(jié)碼以后是根本不存在泛型類型的,甚至是在編譯期就會被抹去,說來說去好像并沒有將泛型擦除說的很透徹,下面我們就以例子的方式來一步一步證明

  • 通過反射驗證編譯期泛型類型被擦除
    1. class Demo1 { 
    2.     public static void main(String[] args) throws Exception { 
    3.         List<Integer> list = new ArrayList<>(); 
    4.         //到這里是沒有任何問題的,正常的一個 集合類的添加元素 
    5.         list.add(1024); 
    6.         list.forEach(System.out::println); 
    7.         System.out.println("-------通過反射證明泛型類型編譯期間被擦除-------"); 
    8.         //反射看不明白的小伙伴不要急,如果想看發(fā)射的文章,請留言反射,我下期保證完成 
    9.         list.getClass().getMethod("add", Object.class).invoke(list, "9527"); 
    10.         for (int i = 0; i < list.size(); i++) { 
    11.             System.out.println("value = " + list.get(i)); 
    12.         } 
    13.     } 

 

打印結(jié)果如下:

 

但是直接同一個反射似乎并不能讓小伙伴們買賬,我們?yōu)榱梭w驗差異,繼續(xù)寫一個例子

  1. class Demo1 { 
  2.     public static void main(String[] args) throws Exception { 
  3.         //List<E>  實際上就是一個泛型,所以我們就不去自己另外寫泛型類來測試了 
  4.         List<Integer> list = new ArrayList<>(); 
  5.         //到這里是沒有任何問題的,正常的一個 集合類的添加元素 
  6.         list.add(1024); 
  7.         list.forEach(System.out::println); 
  8.         System.out.println("-------通過反射證明泛型類型編譯期間被擦除-------"); 
  9.         list.getClass().getMethod("add", Object.class).invoke(list, "9527"); 
  10.         for (int i = 0; i < list.size(); i++) { 
  11.             System.out.println("value = " + list.get(i)); 
  12.         } 
  13.  
  14.         //普通的類 
  15.         FanShe fanShe = new FanShe(); 
  16.         //先通過正常的方式為屬性設(shè)置值 
  17.         fanShe.setStr(1111); 
  18.         System.out.println(fanShe.getStr()); 
  19.         //然后通過同樣的方式為屬性設(shè)置值 不要忘記上面的List  是 List<E> 是泛型哦!不要連最基本的知識都忘記了 
  20.         fanShe.getClass().getMethod("setStr", Object.class).invoke(list, "2222"); 
  21.         System.out.println(fanShe.getStr()); 
  22.     } 
  23. //隨便寫一個類 
  24. class FanShe{ 
  25.     private Integer str; 
  26.     public void setStr(Integer str) { 
  27.         this.str = str; 
  28.     } 
  29.  
  30.     public Integer getStr() { 
  31.         return str; 
  32.     } 

 

測試結(jié)果顯而易見,不是泛型的類型是不能通過反射去修改類型賦值的。

  • 由于泛型擦除帶來的自動類型轉(zhuǎn)換

因為泛型的類型擦除問題,導(dǎo)致所有的泛型類型變量被編譯后都會被替換為原始類型。既然都被替換為原始類型,那么為什么我們在獲取的時候,為什么不需要強制類型轉(zhuǎn)換?

 

下面這么些才是一個標(biāo)準(zhǔn)的帶有泛型返回值的方法。

  1. public class TypeConvert { 
  2.     public static void main(String[] args) { 
  3.  
  4.         //調(diào)用方法的時候返回值就是我們實際傳的泛型的類型 
  5.         MyClazz1 myClazz1 = testTypeConvert(MyClazz1.class); 
  6.         MyClazz2 myClazz2 = testTypeConvert(MyClazz2.class); 
  7.     } 
  8.     private static <T> T testTypeConvert(Class<T> tClass){ 
  9.         //只需要將返回值類型轉(zhuǎn)成實際的泛型類型 T 即可 
  10.         return (T) tClass; 
  11.     } 
  12.  
  13. class MyClazz1{} 
  14. class MyClazz2{} 
  • 由泛型引發(fā)的數(shù)組問題

名字怪嚇人的,實際上說白了就是不能創(chuàng)建泛型數(shù)組

 

看下面的代碼

 

為什么不能創(chuàng)建泛型類型的數(shù)組?

因為List 和 List 被編譯后在 JVM 中等同于List ,所有的類型信息在編譯后都等同于List,也就是說編譯器此時也是無法區(qū)分?jǐn)?shù)組中的具體類型是 Integer類型還是 String 。

但是,使用通配符卻是可以的,我上文還特意強調(diào)過一句話:泛型和通配符不是一回事。請看代碼

 

那這又是為什么?? 表示未知的類型,他的操作不涉及任何的類型相關(guān)的東西,所以 JVM 是不會對其進(jìn)行類型判斷的,因此它能編譯通過,但是這種方式只能讀不能寫,也即只能使用 get 方法,無法使用 add 方法。

為什么不能 add ? 提供了只讀的功能,也就是它刪減了增加具體類型元素的能力,只保留與具體類型無關(guān)的功能。它不管裝載在這個容器內(nèi)的元素是什么類型,它只關(guān)心元素的數(shù)量、容器是否為空,另外上面也已經(jīng)解釋過為什么不能 add 的,這里就當(dāng)做一個補充。

好了,關(guān)于泛型知識,今天就聊到這里,感謝大家的支持!

本文轉(zhuǎn)載自微信公眾號「程序員小灰」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系程序員小灰公眾號。

 

責(zé)任編輯:武曉燕 來源: 程序員小灰
相關(guān)推薦

2024-10-14 08:31:41

泛型策略模式

2020-09-03 11:04:20

Spring 循環(huán)依賴

2022-01-05 07:07:37

Go核心設(shè)計

2009-09-09 14:11:58

Scala泛型

2021-03-01 07:34:42

Java泛型ArrayList

2024-11-29 08:53:46

2021-12-01 09:31:13

安全

2022-01-03 18:07:56

泛型場景demo

2021-10-17 13:10:56

函數(shù)TypeScript泛型

2014-04-15 16:01:00

Java8泛型

2024-04-23 08:23:36

TypeScript泛型Generics

2009-12-24 09:16:11

C#泛型

2019-12-24 10:19:44

泛型反射注解

2021-01-14 05:20:48

Go語言泛型

2009-09-25 10:03:51

Java泛型

2021-06-17 06:51:32

Java泛型Java編程

2024-10-21 07:05:14

C#特性語言

2017-07-14 13:07:03

大數(shù)據(jù)用戶畫像

2017-03-06 16:51:52

Java泛型實現(xiàn)

2024-01-19 08:25:38

死鎖Java通信
點贊
收藏

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

亚洲精品伦理在线| 手机精品视频在线观看| 欧美成人在线直播| 免费看一级大黄情大片| 国产午夜精品一区理论片| 久久一区激情| 欧美成人一区二区三区电影| www国产视频| 韩国av一区二区三区在线观看| 国产一区二区三区av电影 | 亚洲欧洲另类国产综合| 7878成人国产在线观看| 男人天堂av片| av网页在线| 成人午夜私人影院| 国产精品99久久久久久白浆小说| 国产精品白丝喷水在线观看| 日韩动漫一区| 欧美一级夜夜爽| 99久久国产宗和精品1上映| а√天堂资源地址在线下载| 久久综合网色—综合色88| 91精品在线国产| 成人av网站在线播放| 欧美日韩免费| 日韩一区二区欧美| 国产成人精品无码免费看夜聊软件| 国产色99精品9i| 欧美中文字幕一区二区三区亚洲| 男人c女人视频| 欧美三级黄网| 国产欧美中文在线| 六月婷婷久久| 天堂中文在线观看视频| 国产乱码精品一品二品| 国产精品免费视频xxxx| 影音先锋在线国产| 在线日韩视频| 欧美激情在线播放| 天天做夜夜爱爱爱| 日本一区二区三区视频| 亚洲欧洲日产国码av系列天堂| 在线中文字日产幕| 欧美中文高清| 欧美一卡2卡3卡4卡| 亚洲欧美日本一区二区| 日本免费在线一区| 欧洲av在线精品| 日本成人中文字幕在线| 偷拍自拍在线看| 天天色天天操综合| 成人午夜精品久久久久久久蜜臀| 日韩经典av| 亚洲综合一二三区| 91亚洲精品国产| 污的网站在线观看| 亚洲一区二区欧美| 日韩精品在线视频免费观看| 亚洲精品天堂| 亚洲一区二区影院| 国产手机免费视频| 高清视频在线观看三级| 欧美性猛交xxxx乱大交3| 日本毛片在线免费观看| 日韩成人影音| 欧美日韩精品一区二区三区蜜桃 | 干日本少妇视频| 成人无遮挡免费网站视频在线观看 | av天堂一区二区三区| 高清欧美精品xxxxx| 美女视频久久黄| 久久久久青草大香线综合精品| 国产亚洲精品美女久久| 西西人体44www大胆无码| 亚洲国产精品狼友在线观看| 国产人妻精品午夜福利免费| 成人精品一区| 国产精品日日摸夜夜摸av| 亚洲精品中文字幕乱码三区不卡| aaa日本高清在线播放免费观看| 中文成人综合网| 特级西西人体www高清大胆| 国产第一页在线视频| 午夜国产不卡在线观看视频| 男人操女人免费软件| 78精品国产综合久久香蕉| 7777精品久久久大香线蕉| 丝袜熟女一区二区三区| 夜色77av精品影院| 久久精品成人欧美大片| 国产精品99无码一区二区| 久久久久网站| 91免费看网站| 日本一区二区三区在线观看视频| 国产精品久久影院| 国产a级片网站| 成人国产一区二区三区精品麻豆| 日韩女优av电影在线观看| 在线不卡av电影| 在线精品国产| 日韩免费av在线| 国产免费的av| 久久午夜色播影院免费高清 | 人妖一区二区三区| 色黄久久久久久| 97免费在线观看视频| 老汉av免费一区二区三区| 国产美女精品在线观看| 在线免费av网站| 欧美日韩人人澡狠狠躁视频| 五月激情婷婷在线| 亚洲人成网www| 欧美激情第三页| 亚洲图片小说视频| 99精品视频一区| 日本黄网站色大片免费观看| 3d欧美精品动漫xxxx无尽| 精品欧美一区二区三区精品久久| 中文字幕在线观看免费高清| 亚洲人成毛片在线播放女女| 成人免费网站在线观看| 国产高清一级毛片在线不卡| 亚洲成人免费视频| 久久久久亚洲av无码麻豆| 成人3d动漫在线观看| 性色av一区二区三区红粉影视| 国产又黄又爽视频| 中文乱码免费一区二区 | 久久99国产精品免费| 欧美日韩一区二| 美女网站在线看| 精品日韩在线观看| 午夜精品福利在线视频| 麻豆国产精品一区二区三区 | 欧美丝袜一区二区| 少妇高潮一69aⅹ| 亚洲xxx拳头交| 国产欧美精品一区二区三区-老狼 国产欧美精品一区二区三区介绍 国产欧美精品一区二区 | 精品欧美日韩精品| 欧美精品一区二区三区蜜桃视频 | 夜夜嗨一区二区三区| 国产99午夜精品一区二区三区| 午夜毛片在线| 在线看不卡av| 国产一区二区三区四区五区六区 | 一区二区三区在线高清| 亚洲自拍第三页| 亚洲色图网站| 成人性生交大片免费观看嘿嘿视频| 9色在线视频网站| 欧美少妇xxx| 最新日韩免费视频| 久热成人在线视频| 日本成人性视频| 久久丁香四色| 欧美肥老妇视频| 日本精品999| 欧美日韩激情网| 亚洲熟妇一区二区三区| 久久国产主播| 亚洲成人a**址| 91精品网站在线观看| 欧美另类老女人| 亚洲精品18p| 狠狠干狠狠久久| 亚洲精品午夜视频| 久久精品噜噜噜成人av农村| 亚洲综合视频一区| 亚洲日本视频在线| 7m第一福利500精品视频| 黄色片在线免费观看| 欧美三区在线视频| 91精品国产高清一区二区三蜜臀| 国产69精品久久99不卡| 国产精品宾馆在线精品酒店| blacked蜜桃精品一区| 亚洲free性xxxx护士白浆| 草美女在线观看| 亚洲天堂免费在线| 国产乱淫a∨片免费视频| 亚洲韩国一区二区三区| 成人片黄网站色大片免费毛片| 欧美a一区二区| 欧美性视频精品| 精品av中文字幕在线毛片| 欧美日韩国产大片| 麻豆一区产品精品蜜桃的特点| 9l国产精品久久久久麻豆| 毛片毛片毛片毛片毛片毛片毛片毛片毛片 | 韩日精品视频| 韩国精品一区二区三区六区色诱| 欧美亚洲日本精品| 精品国产欧美一区二区三区成人| 亚洲AV无码精品色毛片浪潮| 亚洲综合成人在线| 亚洲一区视频在线播放| 国产乱子轮精品视频| 黄色动漫在线免费看| 91av精品| 日本午夜精品电影| 成人午夜三级| 国产日韩在线精品av| 国产在线美女| 美女av一区二区三区| 黄色毛片在线看| 欧美变态tickling挠脚心| 无码人妻久久一区二区三区| 亚洲色图欧美在线| 国产 欧美 在线| 国产91精品露脸国语对白| 亚洲精品性视频| 久久福利一区| 亚洲国产成人精品无码区99| 久久久久久久久丰满| 欧美日韩在线一二三| 97久久综合精品久久久综合| 国产日韩欧美在线| 日韩性xxx| 91精品国产91久久久久久不卡 | 亚洲丝袜精品丝袜在线| 最新中文字幕视频| 成人三级在线视频| 永久免费黄色片| 青青草国产精品亚洲专区无| 草草久久久无码国产专区| 欧美激情1区2区| 这里只有精品66| 成人av资源电影网站| 免费精品视频一区| 青青草这里只有精品| 国产98在线|日韩| 欧美国产亚洲精品| 92看片淫黄大片欧美看国产片| 欧美性片在线观看| 日韩免费中文字幕| 日韩欧美精品电影| 国产成人午夜视频网址| 在线最新版中文在线| 91福利视频网| 亚洲美女尤物影院| 奇米4444一区二区三区| 免费v片在线观看| 97热在线精品视频在线观看| 大黄网站在线观看| 欧美黑人一区二区三区| 欧美人与牲禽动交com | 日本福利视频导航| 久久久国产精品| 国产香蕉一区二区三区| 影音先锋日韩在线| 伊人再见免费在线观看高清版 | 欧美理伦片在线播放| 国产一区精品视频| 色老板在线视频一区二区| 久久精品日韩精品| 久草成人资源| 先锋影音一区二区三区| 日韩在线第七页| 国产成人三级视频| 国内精品久久久久久久影视蜜臀 | 欧美日韩中文国产一区发布| 神马久久一区二区三区| 图片区小说区区亚洲五月| 91青青国产在线观看精品| 成年人免费观看的视频| 中文字幕一区二区精品区| 日韩精品综合在线| 国产亚洲激情| 色七七在线观看| 国产一区二区三区四区五区美女| 老司机av网站| 久久久久久久免费视频了| 日韩视频在线观看免费视频| 中文字幕亚洲区| 欧洲猛交xxxx乱大交3| 午夜精品福利视频网站| 欧美三级网站在线观看| 欧美一区二视频| 人妻少妇一区二区三区| 一本一本久久a久久精品综合小说| 婷婷在线视频观看| 久久久久久91| 神马电影网我不卡| 91在线精品播放| 免费看成人吃奶视频在线| 黄色高清视频网站| 日韩亚洲精品在线| 在线观看av网页| 成人免费视频视频| 粉嫩精品久久99综合一区| 一区二区三区国产豹纹内裤在线| 一级片中文字幕| 欧美日本一道本在线视频| 天天av天天翘| 精品国产一区二区三区久久久狼 | 91精品国产91久久久久青草| 香蕉视频一区二区三区| 一区二区三视频| 免费亚洲一区| 超碰人人cao| 国产精品污污网站在线观看| 国产在线一二区| 欧美日韩亚洲丝袜制服| 日日夜夜精品免费| 久久久999精品| 日韩不卡免费高清视频| 国产成人精品日本亚洲11| 成人激情开心网| 日韩伦理在线免费观看| 韩日欧美一区二区三区| 人妻体内射精一区二区| 一区二区三区精品在线观看| 亚洲天堂网视频| 亚洲精选中文字幕| 国产又色又爽又黄刺激在线视频| 国产精品日韩电影| 亚洲美女久久| 精品少妇人妻av免费久久洗澡| 精品一区二区三区免费观看| b站大片免费直播| 香蕉成人伊视频在线观看| 国产精品色综合| 曰本色欧美视频在线| 91精品论坛| 国产一区二区不卡视频| 欧美视频一区| aaa一级黄色片| 日韩理论片在线| 亚洲最新av网站| 深夜精品寂寞黄网站在线观看| 在线手机中文字幕| 看高清中日韩色视频| 亚洲国产精品一区| 在线看黄色的网站| 亚洲综合久久av| 亚洲AV无码精品国产| 欧美华人在线视频| 成人性生交大片免费看96| 91看片淫黄大片91| 国产麻豆一精品一av一免费 | 国产精品嫩草69影院| 伊人婷婷欧美激情| 精品人妻伦一区二区三区久久 | 水蜜桃一区二区| 日韩高清一区二区| a级在线免费观看| 欧美影院精品一区| av免费观看一区二区| 国产精品美女www| 久久精品高清| 亚洲精品20p| 一区二区三区四区精品在线视频| 国产乱色精品成人免费视频 | 久久久久久天堂| 精品福利一区二区三区 | 好看的av在线不卡观看| 少妇极品熟妇人妻无码| 亚洲v精品v日韩v欧美v专区| 黄色小视频免费观看| 97超碰蝌蚪网人人做人人爽 | 亚洲美女自拍视频| 日韩国产网站| eeuss中文| 高清成人免费视频| 色网站在线播放| 亚洲色图25p| 中文成人在线| a级黄色片免费| 成人av在线影院| 日韩一级在线视频| 日韩中文字幕不卡视频| 麻豆久久一区| 男女高潮又爽又黄又无遮挡| 国产欧美日韩亚州综合| 99在线精品视频免费观看软件 | 久久国产视频播放| 国产亚洲欧美一区| 国内不卡的一区二区三区中文字幕| 欧美另类videosbestsex日本| 99久久精品免费看国产免费软件| 狠狠人妻久久久久久| 日韩亚洲综合在线| 激情小说亚洲色图| 手机看片福利日韩| 亚洲午夜三级在线| 国产三级在线观看| 99九九视频| 日韩av网站在线观看| 青青草原在线免费观看| 亚洲人成电影网站| 天堂av一区| www.xxx亚洲| 亚洲高清在线精品| 五月婷婷在线观看| 欧美成人第一区| 国产精品综合一区二区三区| 欧美a∨亚洲欧美亚洲| 久久久av电影| 欧美日韩123| 中文字幕人妻一区二区三区|