Java 中的

Jdk5.0新特性Generic Types (泛型)2007-04-27 PChome.net 類型: 轉載 來源: 中國IT實驗室 做者: 未知 責編: 寶良 1. 介紹2.定義簡單Java泛型  其實Java的泛型就是建立一個用類型做爲參數的類。就象咱們寫類的方法同樣,方法是這樣的method(String str1,String str2 ),方法中參數str一、str2的值是可變的。而泛型也是同樣的,這樣寫class Java_Generics<K,V>,這裏邊的K和V就象方法中的參數str1和str2,也是可變。下面看看例子: import java.util.Hashtable; class TestGen0<K,V>{   public Hashtable<K,V> h=new Hashtable<K,V>();   public void put(K k, V v) {    h.put(k,v);   }   public V get(K k) {    return h.get(k);   }   public static void main(String args[]){    TestGen0<String,String> t=new TestGen0<String,String>();    t.put("key", "value");    String s=t.get("key");    System.out.println(s);   } } 正確輸出:value   這只是個例子,不過看看是否是建立一個用類型做爲參數的類,參數是K,V,傳入的「值」是String類型。這個類他沒有特定的待處理型別,之前咱們定義好了一個類,在輸入參數有所固定,是什麼型別的有要求,可是如今編寫程序,徹底能夠不制定參數的類型,具體用的時候來肯定,增長了程序的通用性,像是一個模板。3. 泛型通配符首先,下面是一個例子,做用是打印出一個集合中的全部元素,咱們首先用老版本jdk1.4的編碼規則,代碼以下: void printColleciton(Collection c){ iterator i = c.iterator(); for (k = 0; k < c.size();k++){ System.out.pritnln(i.next(); } 而後,咱們用jdk5.0泛型來重寫上面這段代碼(循環的語法是新版本的語法): void printCollection(Colleciton<Object> c){ for(Object e : c){ System.out.print(e); } }       這個新版本並不比老版本的好多少,老版本能夠用任意一種集合類型做爲參數來調用,而新版本僅僅持有Collection<Object>類型,Colleciton<Object>並非任意類型的Collection的超類。     那麼什麼是全部Colleciton類型的超類型呢?它是Collection<?>這樣一個類型,讀做「未知Colleciton」。它的意思是說Colleciton的元素類型能夠匹配任意類型,咱們把它稱做通配符類型,咱們這樣寫:     void printCollection(Colleciton<?> c){        for (Object e: c){           System.out.println(e);         }      }    如今咱們用任意類型的集合來調用它了,須要注意的是內部方法printColleciton(),咱們任能夠從c中來讀出元素,而且這些元素是Object類型,並且是安全的,由於不管集合中是什麼類型,它總包括Object,可是將任意對象加到集合中是不安全的:      Colleciton<?> c = new ArrayList<String>();      c.add(new Object());//編譯時錯誤    因爲咱們不知道c持有的是什麼類型的元素,咱們不能加object到集合中去。add()方法用類型E做爲參數,(集合的元素類型)當真正的參數類型是?的時候,它表明的是一些未知類型。任何傳遞給add()方法的參數必須是這個未知類型的子類型。因爲咱們不知道未知類型,因此咱們傳遞給它任何東西。主要的例外是null,它是每個類型的成員。    另外一方面,假定給一個List<?>,咱們調用get()而且充分利用結果。結果類型是未知類型。可是我老是知道它是一個Object,所以分配一個從get()取出來的結果到一個object的變量是安全的,或者做爲一個參數傳遞到一個須要object類型的地方。3.1有限制的通配符 考慮一個畫圖的應用程序,這個程序可以畫長方形、圓等類型,爲了在程序中表示這樣的圖形,你能夠定義一個類型的層次結構: public abstract class Shape{          public abstract void draw(Canvas c);  } public class Circle extends Shape{          private int x,y,radius;          public void draw(Canvas c){} } public class Rectangle extends Shape{          private int x,y,width,height;          public void draw(Canvas c){ } } //這些類能被畫在畫布上: public class Canvas{        public void draw(Shape s){                 s.draw(this);        } } 任何畫圖的動做的都包含一些圖形,假設他們被表示在一個list中,在Canvas中它將會有一個很方便的方法來畫他們:    public void drawAll(List<Shape> shapes){              for(Shape s :shapes){                s.draw(this);               }    }     如今類型規則說,方法drawAll()只能在真正的Shape類型的List上被調用,而它的子類沒法調用,例如List<Circle>上被調用。這是很不幸的。因爲全部的方法確實從List中讀出Shape,因此它僅能在List<Object>上被調用,下面咱們改後的代碼能夠在任意類型的Shape上被調用: public void drawAll(List< ? extends Shape>{ } 這裏有一個很小的不一樣是,咱們已經用List<? extends Shape>替換了List<Object>,如今drawAll()方法能夠接受任意的Shape的子類了,咱們固然能夠在List<Circle>上調用。      <? extends Class>是一種限制通配符類型,它能夠接受全部<Class>以及Class的子類型。然而調用代價是,只讀訪問,沒法向shapes中添加元素。像一般同樣,使用通配符帶來的靈活性將付出代價,例如,下面是不容許的:    public void addRectangle(List<? extends Shape> shapes){      shapes.add(0,new Rectangle());//編譯時錯誤    }     限制性通配符的一個例子是,是一我的口普查的例子,咱們假設數據是由一個名字映射一我的,名字是字符串,人(能夠是Person,或是它的子類Driver),Map<k,v>是一個泛型的例子,它擁有兩個參數,表示爲一個KEY和value的映射MAP    再次注意正規參數的命名規則,K表明key,V表明value      public class Census{         public static void addRegistry(Map<String ? extends Person> Registry){ }      }     Map<String,Driver>  allDrivers =;     census.addResigtry(allDrivers); 編寫泛型類要注意:    1) 在定義一個泛型類的時候,在 「<>」之間定義形式類型參數,例如:「class TestGen<K,V>」,其中「K」 , 「V」不表明值,而是表示類型。    2) 實例化泛型對象的時候,必定要在類名後面指定類型參數的值(類型),一共要有兩次書寫。例如:TestGen<String,String> t=new TestGen<String,String>();    3) 泛型中<K extends Object>,extends並不表明繼承,它是類型範圍限制。 4.泛型與數據類型轉換4.1. 消除類型轉換  上面的例子你們看到什麼了,數據類型轉換的代碼不見了。在之前咱們常常要書寫如下代碼,如: import Java.util.Hashtable; class Test {   public static void main(String[] args) {    Hashtable h = new Hashtable();    h.put("key", "value");    String s = (String)h.get("key");    System.out.println(s);   } }   這個咱們作了類型轉換,是否是感受很煩的,而且強制類型轉換會帶來潛在的危險,系統可能會拋一個ClassCastException異常信息。在JDK5.0中咱們徹底能夠這麼作,如: import Java.util.Hashtable; class Test {   public static void main(String[] args) {    Hashtable<String,Integer> h = new Hashtable<String,Integer> ();    h.put("key", new Integer(123));    int s = h.get("key").intValue();    System.out.println(s);   } }   這裏咱們使用泛化版本的HashMap,這樣就不用咱們來編寫類型轉換的代碼了,類型轉換的過程交給編譯器來處理,是否是很方便,並且很安全。上面是String映射到String,也能夠將Integer映射爲String,只要寫成HashTable<Integer,String> h=new HashTable<Integer,String>();h.get(new Integer(0))返回value。果真很方便。 4.2 自動解包裝與自動包裝的功能  從上面有沒有看到有點彆扭啊,h.get(new Integer(123))這裏的new Integer(123);好煩的,在JDK5.0以前咱們只能忍着了,如今這種問題已經解決了,請看下面這個方法。咱們傳入一個int這一基本型別,而後再將i的值直接添加到List中,其實List是不能儲存基本型別的,List中應該存儲對象,這裏編譯器將int包裝成Integer,而後添加到List中去。接着咱們用List.get(0);來檢索數據,並返回對象再將對象解包裝成int。恩,JDK5.0給咱們帶來更多方便與安全。 public void autoBoxingUnboxing(int i) {   ArrayList<Integer> L= new ArrayList<Integer>();   L.add(i);   int a = L.get(0);   System.out.println("The value of i is " + a); } 4.3 限制泛型中類型參數的範圍  也許你已經發如今TestGen<K,V>這個泛型類,其中K,V能夠是任意的型別。也許你有時候呢想限定一下K和V固然範圍,怎麼作呢?看看以下的代碼: class TestGen2<K extents String,V extends Number> {   private V v=null;   private K k=null;   public void setV(V v){    this.v=v;   }   public V getV(){    return this.v;   }   public void setK(K k){    this.k=k;   }   public V getK(){    return this.k;   }   public static void main(String[] args)   {    TestGen2<String,Integer> t2=new TestGen2<String,Integer>();    t2.setK(new String("String"));    t2.setV(new Integer(123));    System.out.println(t2.getK());    System.out.println(t2.getV());   } }   上邊K的範圍是<=String ,V的範圍是<=Number,注意是「<=」,對於K能夠是String的,V固然也能夠是Number,也能夠是Integer,Float,Double,Byte等。看看下圖也許能直觀些請看上圖A是上圖類中的基類,A1,A2分別是A的子類,A2有2個子類分別是A2_1,A2_2。   而後咱們定義一個受限的泛型類class MyGen<E extends A2>,這個泛型的範圍就是上圖中蘭色部分。   這個是單一的限制,你也能夠對型別多重限制,以下: class C<T extends Comparable<? super T> & Serializable>  咱們來分析如下這句,T extends Comparable這個是對上限的限制,Comparable< super T>這個是下限的限制,Serializable是第2個上限。一個指定的類型參數能夠具備一個或多個上限。具備多重限制的類型參數能夠用於訪問它的每一個限制的方法和域。 5.泛型方法 考慮寫一個持有數組類型對象和一個集合對象的方法,把數組裏的全部對象都放到 集合裏。第一個程序爲:   static void fromArrayToColleciton(Object[]a,Collection<?> c){      for (Object o : a){         c.add(o);//編譯時錯誤         }    } 到如今爲止,你可能學會避免開始的錯誤而去使用Collection<Object>做爲集合參數的類型,你可能會意識到使用Colleciton<?>將不會工做。 解決這個問題的方法是使用泛型方法,GENERIC METHODS,就像類型聲明、方法聲明同樣,就是被一個或更多的類型參數參數化。    static <T> void fromArrayToCollection(T[]a,Collection<T> c){         for(T o :a){              c.add(o);//正確             }    }     咱們能夠用任意類型的集合調用這個方法,他的元素類型是數組元素類型的超類型。    Object[] oa = new Object[100];    Collection <Object> co = new ArrayList<Object>();    fromArrayToCollection(oa,co);//T 被認爲是Object類型    String[] sa = new String[100];    Colleciton<String> cs = new ArrayList<String>();    fromArrayToCollection(sa,cs);//T被認爲是String類型    fromArrayToCollection(sa,co);//T 被認爲是Object類型    Integer[] is = new Integer[100];    Float[] fa = new Float[100];    Number[] na = new Number[100];    Collection<Number> cn = new ArrayList<Number>();     fromArrayToCollection(is,cn);//Number    fromArrayToCollection(fa,cn);//Number    fromArrayToCollection(na,cn);//Number    fromArrayToCollection(na,co);//Object    fromArrayToCollection(na,cs);//編譯時錯誤 咱們沒必要給一個泛型方法傳遞一個真正的類型參數,編譯器會推斷類型參數.一個問題出現了,何時使用泛型方法,何時使通配符類型,爲了回答這些問題,咱們從Colleciton庫中看一下幾個方法: interface Collection<E>{            public boolean containsAll(Collection<?> c);            public boolean addAll(Collection<? extends E> c);     }     使用泛型方法的形式爲:     interface Collection<E>{            public <T> boolean containsAll(Collection<T> c);            public <T extends E> boolean addAll(Collection<T> c);     }     不管如何,在ContainAll和addAll中,類型參數T僅被使用一次。返回類型不依賴於類型參數,也不依賴方法中的任何參數。這告訴我類型參數正被用於多態,它的影響僅僅是容許不一樣的實參在不一樣的調用點被使用。    泛型方法容許類型參數被用於表達在一個或更多參數之間或者方法中的參數、返回類型的依賴。若是沒有如此依賴,泛型方法就不能被使用。可能一前一後來聯合使用泛型和通配符,這裏有個例子:     class Collections{       public static <T> void copy(List<T> dest,List<? extends T> src){     }  }     注意兩個參數之間的依賴,任何從原list的對象複製,必須分配一個目標LIST元素的類型T,因而Src的元素類型多是任何T的子類型。咱們沒必要在乎在COPY的表達中,表示依賴使用一個類型參數,可是是使用一個通配符。     下面咱們不使用通配符來重寫上面的方法:      class Collections{       public static <T,S extends T>       void copy(List<T> dest,List<S> src){     }  }      這很是好,可是第一個類型參數既在dst中使用,也在第二個類型參數中使用,S自己就被使用了一次。在類型src中,沒有什麼類型依賴它。這是一個標誌咱們能夠用通配符來替換它。使用通配符比顯示的聲明類型參數更加清楚和精確。因此有可能推薦使用通配符。    通配符也有優點,能夠在方法以外來使用,做爲字段類型、局部變量和數組。 這裏有一個例子。    返回到咱們畫圖的例子,假設咱們要保持一個畫圖請求的歷史,咱們能夠在Shape類內部用一個靜態變量來保持歷史。用drawAll()存儲它到來的參數到歷史字段。    static List<List<? extends Shape>> history =    new ArrayList<List<? extends Shape>>();    public void drawAll(List<? extends Shape> shapes){     history.addLast(shapes);     for (Shape s : shapes){            s.draw(this);          }    }
相關文章
相關標籤/搜索