java-泛型

  1. 泛型
    1. 定義泛型類   public class  XX<T>
    2. 定義泛型接口 public Interface XX<T>
    3. 定義泛型方法 public <T> T fun(T t)
    4. 實例化泛型類   XX<T> t=new XX(String)
    5. 調用泛型方法  fun(String t)
    6. 實例化泛型類型數組
      1. 不能建立一個確切的泛型類型的數組
        //List<String>[] ls = new ArrayList<String>[10];  //錯誤
        List<?>[] ls = new ArrayList<?>[10]; //OK
        List<String>[] ls = new ArrayList[10]; //OK

         

  2. 什麼是泛型
    1. 泛型,即「參數化類型」顧名思義,就是將類型由原來的具體的類型參數化
    2. 泛型的本質是爲了參數化類型(在不建立新的類型的狀況下,經過泛型指定的不一樣類型來控制形參具體限制的類型)。也就是說在泛型使用過程當中,

      操做的數據類型被指定爲一個參數,這種參數類型能夠用在類、接口和方法中
  3. 分類:泛型類、泛型接口、泛型方法
  4. 一個栗子
    1. List arrayList = new ArrayList();
      arrayList.add("aaaa");
      arrayList.add(100);
      
      for(int i = 0; i< arrayList.size();i++){
          String item = (String)arrayList.get(i);
          Log.d("泛型測試","item = " + item);
      }

      毫無疑問,程序的運行結果會以崩潰結束:
      java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
      ArrayList能夠存聽任意類型
       

       

    2. 例子中添加了一個String類型,添加了一個Integer類型,再使用時都以String的方式使用,所以程序崩潰了。爲了解決相似這樣的問題(在編譯階段就能夠解決),泛型應運而生。
    3. 咱們將第一行聲明初始化list的代碼更改一下,編譯器會在編譯階段就可以幫咱們發現相似這樣的問題。html

    4. List<String> arrayList = new ArrayList<String>();
      ...
      //arrayList.add(100); 在編譯階段,編譯器就會報錯
  5. 特性
    1. 泛型只在編譯階段有效
    2. List<String> stringArrayList = new ArrayList<String>();
      List<Integer> integerArrayList = new ArrayList<Integer>();
      
      Class classStringArrayList = stringArrayList.getClass();
      Class classIntegerArrayList = integerArrayList.getClass();
      
      if(classStringArrayList.equals(classIntegerArrayList)){
          Log.d("泛型測試","類型相同");
      }
      
      輸出結果:D/泛型測試: 類型相同。
    3. 經過上面的例子能夠證實,在編譯以後程序會採起去泛型化的措施。也就是說Java中的泛型,只在編譯階段有效。在編譯過程當中,正確檢驗泛型結果後,會將泛型的相關信息擦出,而且在對象進入和離開方法的邊界處添加類型檢查和類型轉換的方法。也就是說,泛型信息不會進入到運行時階段。java

      對此總結成一句話:泛型類型在邏輯上看以當作是多個不一樣的類型,實際上都是相同的基本類型。數組

  6.  泛型類安全

    1. 最典型的就是各類容器類,如:List、Set、Map
    2. 一個最普通的泛型類
      1. //此處T能夠隨便寫爲任意標識,常見的如T、E、K、V等形式的參數經常使用於表示泛型
        //在實例化泛型類時,必須指定T的具體類型
        public class Generic<T>{ 
            //key這個成員變量的類型爲T,T的類型由外部指定  
            private T key;
        
            public Generic(T key) { //泛型構造方法形參key的類型也爲T,T的類型由外部指定
                this.key = key;
            }
        
            public T getKey(){ //泛型方法getKey的返回值類型爲T,T的類型由外部指定
                return key;
            }
        }
      2. //泛型的類型參數只能是類類型(包括自定義類),不能是簡單類型
        //傳入的實參類型需與泛型的類型參數類型相同,即爲Integer.
        Generic<Integer> genericInteger = new Generic<Integer>(123456);
        
        //傳入的實參類型需與泛型的類型參數類型相同,即爲String.
        Generic<String> genericString = new Generic<String>("key_vlaue");
        Log.d("泛型測試","key is " + genericInteger.getKey());
        Log.d("泛型測試","key is " + genericString.getKey());
    3. 定義的泛型類,就必定要傳入泛型類型實參麼?並非這樣,在使用泛型的時候若是傳入泛型實參,則會根據傳入的泛型實參作相應的限制,此時泛型纔會起到本應起到的限制做用。若是不傳入泛型類型實參的話,在泛型類中使用泛型的方法或成員變量定義的類型能夠爲任何的類型。
      1. Generic generic = new Generic("111111");
        Generic generic1 = new Generic(4444);
        Generic generic2 = new Generic(55.55);
        Generic generic3 = new Generic(false);

         

         

    4. 注意:
      1.     泛型的類型參數只能是類類型,不能是簡單類型。
        不能對確切的泛型類型使用instanceof操做。以下面的操做是非法的,編譯時會出錯。   
        if(ex_num instanceof Generic<Number>){ }
  7. 泛型接口

    1. 泛型接口與泛型類的定義及使用基本相同。泛型接口常被用在各類類的生產器中
    2. //定義一個泛型接口
      public interface Generator<T> {
          public T next();
      }

       

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

      1. /**
         * 未傳入泛型實參時,與泛型類的定義相同,在聲明類的時候,需將泛型的聲明也一塊兒加到類中
         * 即:class FruitGenerator<T> implements Generator<T>{
         * 若是不聲明泛型,如:class FruitGenerator implements Generator<T>,編譯器會報錯:"Unknown class"
         */
        class FruitGenerator<T> implements Generator<T>{
            @Override
            public T next() {
                return null;
            }
        }

         

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

      1. /**
         * 傳入泛型實參時:
         * 定義一個生產器實現這個接口,雖然咱們只建立了一個泛型接口Generator<T>
         * 可是咱們能夠爲T傳入無數個實參,造成無數種類型的Generator接口。
         * 在實現類實現泛型接口時,如已將泛型類型傳入實參類型,則全部使用泛型的地方都要替換成傳入的實參類型
         * 即:Generator<T>,public T next();中的的T都要替換成傳入的String類型。
         */
        public class FruitGenerator implements Generator<String> {
        
            private String[] fruits = new String[]{"Apple", "Banana", "Pear"};
        
            @Override
            public String next() {
                Random rand = new Random();
                return fruits[rand.nextInt(3)];
            }
        }

         

  8. 泛型通配符(?)

    1. 在函數定義的形參中,儘可能使用通配符?
    2. 在函數調用的實參中,傳遞具體類型
    3. 通配符用在函數的形參中:Generic<?> obj,當函數調用時,才肯定具體的泛型實參
    4. 咱們知道IngeterNumber的一個子類,同時在特性章節中咱們也驗證過Generic<Ingeter>Generic<Number>其實是相同的一種基本類型。那麼問題來了,在使用Generic<Number>做爲形參的方法中,可否使用Generic<Ingeter>的實例傳入呢?在邏輯上相似於Generic<Number>Generic<Ingeter>是否能夠當作具備父子關係的泛型類型呢?
    5. 爲了弄清楚這個問題,咱們使用Generic<T>這個泛型類繼續看下面的例子:
      1. 定義泛型類 Generic<T>
      2. 定義測試類Test,測試類中定義測試方法
        1. public void showKeyValue1(Generic<Number> obj){
              Log.d("泛型測試","key value is " + obj.getKey());
          }

           

      3. 實例化泛型類的兩個實例
        1. Generic<Integer> gInteger = new Generic<Integer>(123);
          Generic<Number> gNumber = new Generic<Number>(456);

           

      4. 調用測試類
        1. showKeyValue(gNumber);
          
          // showKeyValue這個方法編譯器會爲咱們報錯:Generic<java.lang.Integer> 
          // cannot be applied to Generic<java.lang.Number>
          // showKeyValue(gInteger);

           

    6. 經過提示信息咱們能夠看到Generic<Integer>不能被看做爲`Generic<Number>的子類。由此能夠看出:同一種泛型能夠對應多個版本(由於參數類型是不肯定的),不一樣版本的泛型類實例是不兼容的。dom

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

      咱們能夠將上面的方法改一下:函數

      1. public void showKeyValue1(Generic<?> obj){
            Log.d("泛型測試","key value is " + obj.getKey());
        }

         

      2. 類型通配符通常是使用?代替具體的類型實參,注意了,此處’?’是類型實參,而不是類型形參 。重要說三遍!此處’?’是類型實參,而不是類型形參 ! 此處’?’是類型實參,而不是類型形參 !再直白點的意思就是,此處的?和Number、String、Integer同樣都是一種實際的類型,能夠把?當作全部類型的父類。是一種真實的類型。測試

    7. 能夠解決當具體類型不肯定的時候,這個通配符就是 ?  ;當操做類型時,不須要使用類型的具體功能時,只使用Object類中的功能。那麼能夠用 ? 通配符來表未知類型。
  9. 泛型方法

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

    2. 泛型方法的基本介紹
      1.  

        /**
         * 泛型方法的基本介紹
         * @param tClass 傳入的泛型實參
         * @return T 返回值爲T類型
         * 說明:
         *     1)public 與 返回值中間<T>很是重要,能夠理解爲聲明此方法爲泛型方法。
         *     2)只有聲明瞭<T>的方法纔是泛型方法,泛型類中的使用了泛型的成員方法並非泛型方法。
         *     3)<T>代表該方法將使用泛型類型T,此時才能夠在方法中使用泛型類型T。
         *     4)與泛型類的定義同樣,此處T能夠隨便寫爲任意標識,常見的如T、E、K、V等形式的參數經常使用於表示泛型。
         */
        public <T> T genericMethod(Class<T> tClass)throws InstantiationException ,
          IllegalAccessException{
                T instance = tClass.newInstance();
                return instance;
        }

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

         

    3. 泛型方法的基本用法

      1. public class GenericTest {
           //這個類是個泛型類,在上面已經介紹過
           public class Generic<T>{     
                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
             * 這個T能夠出如今這個泛型方法的任意位置.
             * 泛型的數量也能夠爲任意多個 
             *    如:public <T,K> K showKeyName(Generic<T> container){
             *        ...
             *        }
             */
            public <T> T showKeyName(Generic<T> container){
                System.out.println("container key :" + container.getKey());
                //固然這個例子舉的不太合適,只是爲了說明泛型方法的特性。
                T test = container.getKey();
                return test;
            }
        
            //這也不是一個泛型方法,這就是一個普通的方法,只是使用了Generic<Number>這個泛型類作形參而已。
            public void showKeyValue1(Generic<Number> 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>,也代表了這是一個能夠處理泛型的類型的泛型方法。
             * 可是隻聲明瞭泛型類型T,並未聲明泛型類型E,所以編譯器並不知道該如何處理E這個類型。
            public <T> T showKeyName(Generic<E> container){
                ...
            }  
            */
        
            /**
             * 這個方法也是有問題的,編譯器會爲咱們提示錯誤信息:"UnKnown class 'T' "
             * 對於編譯器來講T這個類型並未項目中聲明過,所以編譯也不知道該如何編譯這個類。
             * 因此這也不是一個正確的泛型方法聲明。
            public void showkey(T genericObj){
        
            }
            */
        
            public static void main(String[] args) {
        
        
            }
        }
        class GenerateTest<T>{
                public void show_1(T t){
                    System.out.println(t.toString());
                }
        
                //在泛型類中聲明瞭一個泛型方法,使用泛型E,這種泛型E能夠爲任意類型。能夠類型與T相同,也能夠不一樣。
                //因爲泛型方法在聲明的時候會聲明泛型<E>,所以即便在泛型類中並未聲明泛型,編譯器也可以正確識別泛型方法中識別的泛型。
                public <E> void show_3(E t){
                    System.out.println(t.toString());
                }
        
                //在泛型類中聲明瞭一個泛型方法,使用泛型T,注意這個T是一種全新的類型,能夠與泛型類中聲明的T不是同一種類型。
                public <T> void show_2(T t){
                    System.out.println(t.toString());
                }
            }

         

    4. 泛型方法與可變參數

      1. 再看一個泛型方法和可變參數的例子
      2. public <T> void printMsg( T... args){
            for(T t : args){
                Log.d("泛型測試","t is " + t);
            }
        }
        
        
        
        
        printMsg("111",222,"aaaa","2323.4",55.55);
    5. 泛型方法總結

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

        不管什麼時候,若是你能作到,你就該儘可能使用泛型方法。也就是說,若是使用泛型方法將整個類泛型化,

        那麼就應該使用泛型方法。另外對於一個static的方法而已,沒法訪問泛型類型的參數。

        因此若是static方法要使用泛型能力,就必須使其成爲泛型方法。
  10. 靜態方法與泛型

    1. 靜態方法有一種狀況須要注意一下,那就是在類中的靜態方法使用泛型:靜態方法沒法訪問類上定義的泛型;若是靜態方法操做的引用數據類型不肯定的時候,必需要將泛型定義在方法上。

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

    2.  

      public class StaticGenerator<T> {
          ....
          ....
          /**
           * 若是在類中定義使用泛型的靜態方法,須要添加額外的泛型聲明(將這個方法定義成泛型方法)
           * 即便靜態方法要使用泛型類中已經聲明過的泛型也不能夠。
           * 如:public static void show(T t){..},此時編譯器會提示錯誤信息:
                "StaticGenerator cannot be refrenced from static context"
           */
          public static <T> void show(T t){
      
          }
      }

       

  11. 泛型上下邊界

    1. 在使用泛型的時候,咱們還能夠爲傳入的泛型類型實參進行上下邊界的限制,如:類型實參只准傳入某種類型的父類或某種類型的子類。

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

      public void showKeyValue1(Generic<? extends Number> obj){ Log.d("泛型測試","key value is " + obj.getKey()); }
      複製代碼
      Generic<String> generic1 = new Generic<String>("11111"); Generic<Integer> generic2 = new Generic<Integer>(2222); Generic<Float> generic3 = new Generic<Float>(2.4f); Generic<Double> generic4 = new Generic<Double>(2.56); //這一行代碼編譯器會提示錯誤,由於String類型並非Number類型的子類 //showKeyValue1(generic1);  showKeyValue1(generic2); showKeyValue1(generic3); showKeyValue1(generic4);
      複製代碼

      若是咱們把泛型類的定義也改一下:

      複製代碼
      public class Generic<T extends Number>{ private T key; public Generic(T key) { this.key = key; } public T getKey(){ return key; } }
      複製代碼
      //這一行代碼也會報錯,由於String不是Number的子類 Generic<String> generic1 = new Generic<String>("11111");

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

      複製代碼
      //在泛型方法中添加上下邊界限制的時候,必須在權限聲明與返回值之間的<T>上添加上下邊界,即在泛型聲明的時候添加 //public <T> T showKeyName(Generic<T extends Number> container),編譯器會報錯:"Unexpected bound" public <T extends Number> T showKeyName(Generic<T> container){ System.out.println("container key :" + container.getKey()); T test = container.getKey(); return test; }
      複製代碼

      經過上面的兩個例子能夠看出:泛型的上下邊界添加,必須與泛型的聲明在一塊兒 。

  12.  

    關於泛型數組要提一下
    1. 看到了不少文章中都會提起泛型數組,通過查看sun的說明文檔,在java中是」不能建立一個確切的泛型類型的數組」的。

      也就是說下面的這個例子是不能夠的:

      List<String>[] ls = new ArrayList<String>[10]; 

      而使用通配符建立泛型數組是能夠的,以下面這個例子:

      List<?>[] ls = new ArrayList<?>[10]; 

      這樣也是能夠的:

      List<String>[] ls = new ArrayList[10];

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

      複製代碼
      List<String>[] lsa = new List<String>[10]; // Not really allowed. Object o = lsa; Object[] oa = (Object[]) o; List<Integer> li = new ArrayList<Integer>(); 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<Integer> li = new ArrayList<Integer>(); li.add(new Integer(3)); oa[1] = li; // Correct. Integer i = (Integer) lsa[1].get(0); // OK 
      複製代碼
  13. 參考:https://www.cnblogs.com/coprince/p/8603492.html
相關文章
相關標籤/搜索