泛型濃縮筆記

      網友投稿 661 2022-05-30

      泛型濃縮筆記

      一、?泛型概述

      泛型,即“參數化類型”。一提到參數,最熟悉的就是定義方法時有形參,然后調用此方法時傳遞實參。那么參數化類型怎么理解呢?顧名思義,就是將類型由原來的具體的類型參數化,類似于方法中的變量參數,此時類型也定義成參數形式(可以稱之為類型形參),然后在使用/調用時傳入具體的類型(類型實參)。

      泛型的本質是為了參數化類型(在不創建新的類型的情況下,通過泛型指定的不同類型來控制形參具體限制的類型)。也就是說在泛型使用過程中,操作的數據類型被指定為一個參數,這種參數類型可以用在類、接口和方法中,分別被稱為泛型類、泛型接口、泛型方法。

      二、泛型作用

      泛型的作用是限制類、接口、方法傳入參數的類型。

      List arrayList = new ArrayList();

      arrayList.add("aaaa");

      arrayList.add(100);

      三、泛型類

      泛型類型用于類的定義中,被稱為泛型類。通過泛型可以完成對一組類的操作對外開放相同的接口。最典型的就是各種容器類,如:List、Set、Map。

      一個最普通的泛型類:

      //此處T可以隨便寫為任意標識,常見的如T、E、K、V等形式的參數常用于表示泛型

      //在實例化泛型類時,必須指定T的具體類型

      public class Generic{

      //key這個成員變量的類型為T,T的類型由外部指定

      private T key;

      public Generic(T key) { //泛型構造方法形參key的類型也為T,T的類型由外部指定

      this.key = key;

      }

      public T getKey(){ //泛型方法getKey的返回值類型為T,T的類型由外部指定

      return key;

      }

      }

      //泛型的類型參數只能是類類型(包括自定義類),不能是簡單類型

      //傳入的實參類型需與泛型的類型參數類型相同,即為Integer.

      Generic genericInteger = new Generic(123456);

      //傳入的實參類型需與泛型的類型參數類型相同,即為String.

      Generic genericString = new Generic("key_vlaue");

      Log.d("泛型測試","key is " + genericInteger.getKey());

      Log.d("泛型測試","key is " + genericString.getKey());

      定義的泛型類,就一定要傳入泛型類型實參么?并不是這樣,在使用泛型的時候如果傳入泛型實參,則會根據傳入的泛型實參做相應的限制,此時泛型才會起到本應起到的限制作用。如果不傳入泛型類型實參的話,在泛型類中使用泛型的方法或成員變量定義的類型可以為任何的類型。

      看一個例子:

      Generic generic = new Generic("111111");

      Generic generic1 = new Generic(4444);

      Generic generic2 = new Generic(55.55);

      Generic generic3 = new Generic(false);

      Log.d("泛型測試","key is " + generic.getKey());

      Log.d("泛型測試","key is " + generic1.getKey());

      Log.d("泛型測試","key is " + generic2.getKey());

      Log.d("泛型測試","key is " + generic3.getKey());

      注意:

      泛型的類型參數只能是類類型,不能是簡單類型。

      不能對確切的泛型類型使用instanceof操作。如下面的操作是非法的,編譯時會出錯。

      if(ex_num instanceof Generic){

      }

      四、泛型接口

      泛型接口與泛型類的定義及使用基本相同。泛型接口常被用在各種類的生產器中,可以看一個例子:

      //定義一個泛型接口

      public interface Generator {

      public T next();

      }

      當實現泛型接口的類,未傳入泛型實參時:

      /**

      * 未傳入泛型實參時,與泛型類的定義相同,在聲明類的時候,需將泛型的聲明也一起加到類中

      * 即:class FruitGenerator implements Generator{

      * 如果不聲明泛型,如:class FruitGenerator implements Generator,編譯器會報錯:"Unknown class"

      */

      class FruitGenerator implements Generator{

      @Override

      public T next() {

      return null;

      }

      }

      當實現泛型接口的類,傳入泛型實參時:

      /**

      * 傳入泛型實參時:

      * 定義一個生產器實現這個接口,雖然我們只創建了一個泛型接口Generator

      * 但是我們可以為T傳入無數個實參,形成無數種類型的Generator接口。

      * 在實現類實現泛型接口時,如已將泛型類型傳入實參類型,則所有使用泛型的地方都要替換成傳入的實參類型

      * 即:Generator,public T next();中的的T都要替換成傳入的String類型。

      */

      public class FruitGenerator implements Generator {

      private String[] fruits = new String[]{"Apple", "Banana", "Pear"};

      @Override

      public String next() {

      Random rand = new Random();

      return fruits[rand.nextInt(3)];

      }

      }

      五、泛型通配符

      我們知道Ingeter是Number的一個子類,同時在特性章節中我們也驗證過Generic與Generic實際上是相同的一種基本類型。那么問題來了,在使用Generic作為形參的方法中,能否使用Generic的實例傳入呢?在邏輯上類似于Generic和Generic是否可以看成具有父子關系的泛型類型呢?

      為了弄清楚這個問題,我們使用Generic這個泛型類繼續看下面的例子:

      public void showKeyValue1(Generic obj){

      Log.d("泛型測試","key value is " + obj.getKey());

      }

      Generic gInteger = new Generic(123);

      Generic gNumber = new Generic(456);

      showKeyValue(gNumber);

      // showKeyValue這個方法編譯器會為我們報錯:Generic

      // cannot be applied to Generic

      // showKeyValue(gInteger);

      通過提示信息我們可以看到Generic不能被看作為`Generic的子類。由此可以看出:同一種泛型可以對應多個版本(因為參數類型是不確定的),不同版本的泛型類實例是不兼容的。

      回到上面的例子,如何解決上面的問題?總不能為了定義一個新的方法來處理Generic類型的類,這顯然與java中的多臺理念相違背。因此我們需要一個在邏輯上可以表示同時是Generic和Generic父類的引用類型。由此類型通配符應運而生。

      我們可以將上面的方法改一下:

      public void showKeyValue1(Generic obj){

      Log.d("泛型測試","key value is " + obj.getKey());

      }

      類型通配符一般是使用?代替具體的類型實參,注意了,此處’?’是類型實參,而不是類型形參 。重要說三遍!此處’?’是類型實參,而不是類型形參 ! 此處’?’是類型實參,而不是類型形參 !再直白點的意思就是,此處的?和Number、String、Integer一樣都是一種實際的類型,可以把?看成所有類型的父類。是一種真實的類型。

      可以解決當具體類型不確定的時候,這個通配符就是????;當操作類型時,不需要使用類型的具體功能時,只使用Object類中的功能。那么可以用 ? 通配符來表未知類型。

      六、泛型方法

      在java中,泛型類的定義非常簡單,但是泛型方法就比較復雜了。

      尤其是我們見到的大多數泛型類中的成員方法也都使用了泛型,有的甚至泛型類中也包含著泛型方法,這樣在初學者中非常容易將泛型方法理解錯了。

      泛型類,是在實例化類的時候指明泛型的具體類型;泛型方法,是在調用方法的時候指明泛型的具體類型 。

      /**

      * 泛型方法的基本介紹

      * @param tClass 傳入的泛型實參

      * @return T 返回值為T類型

      * 說明:

      * 1)public 與 返回值中間非常重要,可以理解為聲明此方法為泛型方法。

      * 2)只有聲明了的方法才是泛型方法,泛型類中的使用了泛型的成員方法并不是泛型方法。

      * 3)表明該方法將使用泛型類型T,此時才可以在方法中使用泛型類型T。

      * 4)與泛型類的定義一樣,此處T可以隨便寫為任意標識,常見的如T、E、K、V等形式的參數常用于表示泛型。

      */

      public T genericMethod(Class tClass)throws InstantiationException ,

      IllegalAccessException{

      T instance = tClass.newInstance();

      return instance;

      Object obj = genericMethod(Class.forName("com.test.test"));

      4.6.1 泛型方法的基本用法

      光看上面的例子有的同學可能依然會非常迷糊,我們再通過一個例子,把我泛型方法再總結一下。

      public class GenericTest {

      //這個類是個泛型類,在上面已經介紹過

      public class Generic{

      private T key;

      public Generic(T key) {

      this.key = key;

      }

      //我想說的其實是這個,雖然在方法中使用了泛型,但是這并不是一個泛型方法。

      //這只是類中一個普通的成員方法,只不過他的返回值是在聲明泛型類已經聲明過的泛型。

      //所以在這個方法中才可以繼續使用 T 這個泛型。

      public T getKey(){

      return key;

      }

      /**

      * 這個方法顯然是有問題的,在編譯器會給我們提示這樣的錯誤信息"cannot reslove symbol E"

      * 因為在類的聲明中并未聲明泛型E,所以在使用E做形參和返回值類型時,編譯器會無法識別。

      public E setKey(E key){

      this.key = keu

      }

      */

      }

      /**

      * 這才是一個真正的泛型方法。

      * 首先在public與返回值之間的必不可少,這表明這是一個泛型方法,并且聲明了一個泛型T

      * 這個T可以出現在這個泛型方法的任意位置.

      * 泛型的數量也可以為任意多個

      * 如:public K showKeyName(Generic container){

      * ...

      * }

      */

      public T showKeyName(Generic container){

      System.out.println("container key :" + container.getKey());

      //當然這個例子舉的不太合適,只是為了說明泛型方法的特性。

      泛型濃縮筆記

      T test = container.getKey();

      return test;

      }

      //這也不是一個泛型方法,這就是一個普通的方法,只是使用了Generic這個泛型類做形參而已。

      public void showKeyValue1(Generic obj){

      Log.d("泛型測試","key value is " + obj.getKey());

      }

      //這也不是一個泛型方法,這也是一個普通的方法,只不過使用了泛型通配符?

      //同時這也印證了泛型通配符章節所描述的,?是一種類型實參,可以看做為Number等所有類的父類

      public void showKeyValue2(Generic obj){

      Log.d("泛型測試","key value is " + obj.getKey());

      }

      /**

      * 這個方法是有問題的,編譯器會為我們提示錯誤信息:"UnKnown class 'E' "

      * 雖然我們聲明了,也表明了這是一個可以處理泛型的類型的泛型方法。

      * 但是只聲明了泛型類型T,并未聲明泛型類型E,因此編譯器并不知道該如何處理E這個類型。

      public T showKeyName(Generic container){

      ...

      }

      */

      /**

      * 這個方法也是有問題的,編譯器會為我們提示錯誤信息:"UnKnown class 'T' "

      * 對于編譯器來說T這個類型并未項目中聲明過,因此編譯也不知道該如何編譯這個類。

      * 所以這也不是一個正確的泛型方法聲明。

      public void showkey(T genericObj){

      }

      */

      public static void main(String[] args) {

      }

      }

      4.6.2 類中的泛型方法

      當然這并不是泛型方法的全部,泛型方法可以出現雜任何地方和任何場景中使用。但是有一種情況是非常特殊的,當泛型方法出現在泛型類中時,我們再通過一個例子看一下

      public class GenericFruit {

      class Fruit{

      @Override

      public String toString() {

      return "fruit";

      }

      }

      class Apple extends Fruit{

      @Override

      public String toString() {

      return "apple";

      }

      }

      class Person{

      @Override

      public String toString() {

      return "Person";

      }

      }

      class GenerateTest{

      public void show_1(T t){

      System.out.println(t.toString());

      }

      //在泛型類中聲明了一個泛型方法,使用泛型E,這種泛型E可以為任意類型??梢灶愋团cT相同,也可以不同。

      //由于泛型方法在聲明的時候會聲明泛型,因此即使在泛型類中并未聲明泛型,編譯器也能夠正確識別泛型方法中識別的泛型。

      public void show_3(E t){

      System.out.println(t.toString());

      }

      //在泛型類中聲明了一個泛型方法,使用泛型T,注意這個T是一種全新的類型,可以與泛型類中聲明的T不是同一種類型。

      public void show_2(T t){

      System.out.println(t.toString());

      }

      }

      public static void main(String[] args) {

      Apple apple = new Apple();

      Person person = new Person();

      GenerateTest generateTest = new GenerateTest();

      //apple是Fruit的子類,所以這里可以

      generateTest.show_1(apple);

      //編譯器會報錯,因為泛型類型實參指定的是Fruit,而傳入的實參類是Person

      //generateTest.show_1(person);

      //使用這兩個方法都可以成功

      generateTest.show_2(apple);

      generateTest.show_2(person);

      //使用這兩個方法也都可以成功

      generateTest.show_3(apple);

      generateTest.show_3(person);

      }

      }

      4.6.3 泛型方法與可變參數

      再看一個泛型方法和可變參數的例子:

      public void printMsg( T... args){

      for(T t : args){

      Log.d("泛型測試","t is " + t);

      }

      }

      4.6.4 靜態方法與泛型

      靜態方法有一種情況需要注意一下,那就是在類中的靜態方法使用泛型:靜態方法無法訪問類上定義的泛型;如果靜態方法操作的引用數據類型不確定的時候,必須要將泛型定義在方法上。

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

      public class StaticGenerator {

      ....

      ....

      /**

      * 如果在類中定義使用泛型的靜態方法,需要添加額外的泛型聲明(將這個方法定義成泛型方法)

      * 即使靜態方法要使用泛型類中已經聲明過的泛型也不可以。

      * 如:public static void show(T t){..},此時編譯器會提示錯誤信息:

      "StaticGenerator cannot be refrenced from static context"

      */

      public static void show(T t){

      }

      }

      4.6.5 泛型方法總結

      泛型方法能使方法獨立于類而產生變化,以下是一個基本的指導原則:

      無論何時,如果你能做到,你就該盡量使用泛型方法。也就是說,如果使用泛型方法將整個類泛型化,那么就應該使用泛型方法。另外對于一個static的方法而已,無法訪問泛型類型的參數。所以如果static方法要使用泛型能力,就必須使其成為泛型方法。

      4.6 泛型上下邊界

      在使用泛型的時候,我們還可以為傳入的泛型類型實參進行上下邊界的限制,如:類型實參只準傳入某種類型的父類或某種類型的子類。

      為泛型添加上邊界,即傳入的類型實參必須是指定類型的子類型

      public void showKeyValue1(Generic obj){

      Log.d("泛型測試","key value is " + obj.getKey());

      }

      Generic generic1 = new Generic("11111");

      Generic generic2 = new Generic(2222);

      Generic generic3 = new Generic(2.4f);

      Generic generic4 = new Generic(2.56);

      //這一行代碼編譯器會提示錯誤,因為String類型并不是Number類型的子類

      //showKeyValue1(generic1);

      showKeyValue1(generic2);

      showKeyValue1(generic3);

      showKeyValue1(generic4);

      如果我們把泛型類的定義也改一下:

      public class Generic{

      private T key;

      public Generic(T key) {

      this.key = key;

      }

      public T getKey(){

      return key;

      }

      }

      //這一行代碼也會報錯,因為String不是Number的子類

      //這一行代碼也會報錯,因為String不是Number的子類

      Generic generic1 = new Generic("11111");

      再來一個泛型方法的例子:

      //在泛型方法中添加上下邊界限制的時候,必須在權限聲明與返回值之間的上添加上下邊界,即在泛型聲明的時候添加

      //public T showKeyName(Generic container),編譯器會報錯:"Unexpected bound"

      public T showKeyName(Generic container){

      System.out.println("container key :" + container.getKey());

      T test = container.getKey();

      return test;

      }

      通過上面的兩個例子可以看出:泛型的上下邊界添加,必須與泛型的聲明在一起 。

      4.7 關于泛型數組要提一下

      看到了很多文章中都會提起泛型數組,經過查看sun的說明文檔,在java中是”不能創建一個確切的泛型類型的數組”的。

      也就是說下面的這個例子是不可以的:

      List[] ls = new ArrayList[10];

      1

      而使用通配符創建泛型數組是可以的,如下面這個例子:

      List[] ls = new ArrayList[10];

      1

      這樣也是可以的:

      List[] ls = new ArrayList[10];

      1

      下面使用Sun的一篇文檔的一個例子來說明這個問題:

      List[] lsa = new List[10]; // Not really allowed.

      Object o = lsa;

      Object[] oa = (Object[]) o;

      List li = new ArrayList();

      li.add(new Integer(3));

      oa[1] = li; // Unsound, but passes run time store check

      String s = lsa[1].get(0); // Run-time error: ClassCastException.

      這種情況下,由于JVM泛型的擦除機制,在運行時JVM是不知道泛型信息的,所以可以給oa[1]賦上一個ArrayList而不會出現異常,但是在取出數據的時候卻要做一次類型轉換,所以就會出現ClassCastException,如果可以進行泛型數組的聲明,上面說的這種情況在編譯期將不會出現任何的警告和錯誤,只有在運行時才會出錯。

      而對泛型數組的聲明進行限制,對于這樣的情況,可以在編譯期提示代碼有類型安全問題,比沒有任何提示要強很多。

      下面采用通配符的方式是被允許的:數組的類型不可以是類型變量,除非是采用通配符的方式,因為對于通配符的方式,最后取出數據是要做顯式的類型轉換的。

      List[] lsa = new List[10]; // OK, array of unbounded wildcard type.

      Object o = lsa;

      Object[] oa = (Object[]) o;

      List li = new ArrayList();

      li.add(new Integer(3));

      oa[1] = li; // Correct.

      Integer i = (Integer) lsa[1].get(0); // OK

      C++ 容器 面向對象編程

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      上一篇:Excel2016表格中更換字體的方法有哪些
      下一篇:專訪張鑫旭:潛心鉆研技術喜歡與眾不同成就《CSS世界》
      相關文章
      亚洲jizzjizz少妇| 亚洲第一成年网站视频| 亚洲成av人片天堂网老年人| 亚洲熟妇AV一区二区三区浪潮 | 亚洲中文字幕无码爆乳av中文| 亚洲精品无码aⅴ中文字幕蜜桃| 中日韩亚洲人成无码网站| 自拍日韩亚洲一区在线| 最新国产成人亚洲精品影院| 亚洲精品人成网在线播放影院| 亚洲香蕉久久一区二区三区四区| 亚洲天堂一区在线| 亚洲av永久无码精品天堂久久| 亚洲一区二区三区播放在线| 亚洲一区中文字幕在线电影网| 久久亚洲精品专区蓝色区| 精品亚洲AV无码一区二区| 亚洲欧洲日韩国产一区二区三区| www.亚洲日本| 亚洲色欲啪啪久久WWW综合网| 亚洲精品国产精品| 欧洲亚洲综合一区二区三区 | 亚洲欧美乱色情图片| 亚洲日韩一区二区一无码| 亚洲AV永久无码天堂影院| 色欲aⅴ亚洲情无码AV| 亚洲精品无码av天堂| 激情97综合亚洲色婷婷五| 国产亚洲婷婷香蕉久久精品| 亚洲Av永久无码精品三区在线| 久久亚洲AV无码精品色午夜麻| 久久精品国产亚洲av影院| 亚洲国产精品综合久久2007| 亚洲AV无码无限在线观看不卡| 亚洲欧美日韩一区二区三区在线| 久久亚洲色WWW成人欧美| 亚洲精品在线视频| 亚洲精品卡2卡3卡4卡5卡区| 亚洲高清在线播放| 亚洲av片不卡无码久久| 亚洲成AV人片在WWW|