您好,登錄后才能下訂單哦!
本篇內(nèi)容介紹了“Java中的泛型怎么應(yīng)用”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
泛型在java中有很重要的地位,在面向?qū)ο缶幊碳案鞣N設(shè)計模式中有非常廣泛的應(yīng)用。
什么是泛型?為什么要使用泛型?
泛型,即“參數(shù)化類型”。一提到參數(shù),最熟悉的就是定義方法時有形參,然后調(diào)用此方法時傳遞實參。那么參數(shù)化類型怎么理解呢?顧名思義,就是將類型由原來的具體的類型參數(shù)化,類似于方法中的變量參數(shù),此時類型也定義成參數(shù)形式(可以稱之為類型形參),然后在使用/調(diào)用時傳入具體的類型(類型實參)。
泛型的本質(zhì)是為了參數(shù)化類型(在不創(chuàng)建新的類型的情況下,通過泛型指定的不同類型來控制形參具體限制的類型)。也就是說在泛型使用過程中,操作的數(shù)據(jù)類型被指定為一個參數(shù),這種參數(shù)類型可以用在類、接口和方法中,分別被稱為泛型類、泛型接口、泛型方法。
一個被舉了無數(shù)次的例子:
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); }
毫無疑問,程序的運行結(jié)果會以崩潰結(jié)束:
java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
ArrayList可以存放任意類型,例子中添加了一個String類型,添加了一個Integer類型,再使用時都以String的方式使用,因此程序崩潰了。為了解決類似這樣的問題(在編譯階段就可以解決),泛型應(yīng)運而生。
我們將第一行聲明初始化list的代碼更改一下,編譯器會在編譯階段就能夠幫我們發(fā)現(xiàn)類似這樣的問題。
List
…
//arrayList.add(100); 在編譯階段,編譯器就會報錯
泛型只在編譯階段有效??聪旅娴拇a:
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("泛型測試","類型相同"); }
通過上面的例子可以證明,在編譯之后程序會采取去泛型化的措施。也就是說Java中的泛型,只在編譯階段有效。在編譯過程中,正確檢驗泛型結(jié)果后,會將泛型的相關(guān)信息擦出,并且在對象進(jìn)入和離開方法的邊界處添加類型檢查和類型轉(zhuǎn)換的方法。也就是說,泛型信息不會進(jìn)入到運行時階段。
對此總結(jié)成一句話:泛型類型在邏輯上看以看成是多個不同的類型,實際上都是相同的基本類型。
泛型有三種使用方式,分別為:泛型類、泛型接口、泛型方法
泛型類型用于類的定義中,被稱為泛型類。通過泛型可以完成對一組類的操作對外開放相同的接口。最典型的就是各種容器類,如:List、Set、Map。
泛型類的最基本寫法(這么看可能會有點暈,會在下面的例子中詳解):
class 類名稱 <泛型標(biāo)識:可以隨便寫任意標(biāo)識號,標(biāo)識指定的泛型的類型>{ private 泛型標(biāo)識 /*(成員變量類型)*/ var; ..... }
一個最普通的泛型類:
//此處T可以隨便寫為任意標(biāo)識,常見的如T、E、K、V等形式的參數(shù)常用于表示泛型
//在實例化泛型類時,必須指定T的具體類型 public class Generic<T>{ //在類中聲明的泛型整個類里面都可以用,除了靜態(tài)部分,因為泛型是實例化時聲明的。 //靜態(tài)區(qū)域的代碼在編譯時就已經(jīng)確定,只與類相關(guān) class A <E>{ T t; } //類里面的方法或類中再次聲明同名泛型是允許的,并且該泛型會覆蓋掉父類的同名泛型T class B <T>{ T t; } //靜態(tài)內(nèi)部類也可以使用泛型,實例化時賦予泛型實際類型 static class C <T> { T t; } public static void main(String[] args) { //報錯,不能使用T泛型,因為泛型T屬于實例不屬于類 // T t = null; } //key這個成員變量的類型為T,T的類型由外部指定 private T key; public Generic(T key) { //泛型構(gòu)造方法形參key的類型也為T,T的類型由外部指定 this.key = key; } public T getKey(){ //泛型方法getKey的返回值類型為T,T的類型由外部指定 return key; } }
12-27 09:20:04.432 13063-13063/? D/泛型測試: key is 123456
12-27 09:20:04.432 13063-13063/? D/泛型測試: key is key_vlaue
定義的泛型類,就一定要傳入泛型類型實參么?并不是這樣,在使用泛型的時候如果傳入泛型實參,則會根據(jù)傳入的泛型實參做相應(yīng)的限制,此時泛型才會起到本應(yīng)起到的限制作用。如果不傳入泛型類型實參的話,在泛型類中使用泛型的方法或成員變量定義的類型可以為任何的類型。
看一個例子:
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()); D/泛型測試: key is 111111 D/泛型測試: key is 4444 D/泛型測試: key is 55.55 D/泛型測試: key is false
注意:
泛型的類型參數(shù)只能是類類型,不能是簡單類型。
不能對確切的泛型類型使用instanceof操作。如下面的操作是非法的,編譯時會出錯。
if(ex_num instanceof Generic
}
泛型接口與泛型類的定義及使用基本相同。泛型接口常被用在各種類的生產(chǎn)器中,可以看一個例子:
//定義一個泛型接口 public interface Generator<T> { public T next(); }
當(dāng)實現(xiàn)泛型接口的類,未傳入泛型實參時:
/** * 未傳入泛型實參時,與泛型類的定義相同,在聲明類的時候,需將泛型的聲明也一起加到類中 * 即: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; } }
當(dāng)實現(xiàn)泛型接口的類,傳入泛型實參時:
/** * 傳入泛型實參時: * 定義一個生產(chǎn)器實現(xiàn)這個接口,雖然我們只創(chuàng)建了一個泛型接口Generator<T> * 但是我們可以為T傳入無數(shù)個實參,形成無數(shù)種類型的Generator接口。 * 在實現(xiàn)類實現(xiàn)泛型接口時,如已將泛型類型傳入實參類型,則所有使用泛型的地方都要替換成傳入的實參類型 * 即: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)]; } }
我們知道Ingeter是Number的一個子類,同時在特性章節(jié)中我們也驗證過Generic
為了弄清楚這個問題,我們使用Generic
public void showKeyValue1(Generic<Number> obj){ Log.d("泛型測試","key value is " + obj.getKey()); } Generic<Integer> gInteger = new Generic<Integer>(123); Generic<Number> gNumber = new Generic<Number>(456); showKeyValue(gNumber); // showKeyValue這個方法編譯器會為我們報錯:Generic<java.lang.Integer> // cannot be applied to Generic<java.lang.Number> // showKeyValue(gInteger);
通過提示信息我們可以看到Generic
回到上面的例子,如何解決上面的問題?總不能為了定義一個新的方法來處理Generic
我們可以將上面的方法改一下:
public void showKeyValue1(Generic<?> obj){ Log.d("泛型測試","key value is " + obj.getKey());
類型通配符一般是使用?代替具體的類型實參,注意, 此處的?和Number、String、Integer一樣都是一種實際的類型,可以把?看成所有類型的父類。是一種真實的類型。
可以解決當(dāng)具體類型不確定的時候,這個通配符就是 ? ;當(dāng)操作類型時,不需要使用類型的具體功能時,只使用Object類中的功能。那么可以用 ? 通配符來表未知類型
public void showKeyValue(Generic
System.out.println(obj);
}
Generic<Integer> gInteger = new Generic<Integer>(123); Generic<Number> gNumber = new Generic<Number>(456); public void test () { // showKeyValue(gInteger);該方法會報錯 showKeyValue1(gInteger); } public void showKeyValue1(Generic<?> obj) { System.out.println(obj); } // showKeyValue這個方法編譯器會為我們報錯:Generic<java.lang.Integer> // cannot be applied to Generic<java.lang.Number> // showKeyValue(gInteger);
。
在java中,泛型類的定義非常簡單,但是泛型方法就比較復(fù)雜了。
尤其是我們見到的大多數(shù)泛型類中的成員方法也都使用了泛型,有的甚至泛型類中也包含著泛型方法,這樣在初學(xué)者中非常容易將泛型方法理解錯了。
泛型類,是在實例化類的時候指明泛型的具體類型;泛型方法,是在調(diào)用方法的時候指明泛型的具體類型 。
/** * 泛型方法的基本介紹 * @param tClass 傳入的泛型實參 * @return T 返回值為T類型 * 說明: * 1)public 與 返回值中間<T>非常重要,可以理解為聲明此方法為泛型方法。 * 2)只有聲明了<T>的方法才是泛型方法,泛型類中的使用了泛型的成員方法并不是泛型方法。 * 3)<T>表明該方法將使用泛型類型T,此時才可以在方法中使用泛型類型T。 * 4)與泛型類的定義一樣,此處T可以隨便寫為任意標(biāo)識,常見的如T、E、K、V等形式的參數(shù)常用于表示泛型。 */ public <T> T genericMethod(Class<T> tClass)throws InstantiationException , IllegalAccessException{ T instance = tClass.newInstance(); return instance; } Object obj = genericMethod(Class.forName("com.test.test"));
光看上面的例子有的同學(xué)可能依然會非常迷糊,我們再通過一個例子,把我泛型方法再總結(jié)一下。
/** * 這才是一個真正的泛型方法。 * 首先在public與返回值之間的<T>必不可少,這表明這是一個泛型方法,并且聲明了一個泛型T * 這個T可以出現(xiàn)在這個泛型方法的任意位置. * 泛型的數(shù)量也可以為任意多個 * 如:public <T,K> K showKeyName(Generic<T> container){ * ... * } */ public class 泛型方法 { @Test public void test() { test1(); test2(new Integer(2)); test3(new int[3],new Object()); //打印結(jié)果 // null // 2 // [I@3d8c7aca // java.lang.Object@5ebec15 } //該方法使用泛型T public <T> void test1() { T t = null; System.out.println(t); } //該方法使用泛型T //并且參數(shù)和返回值都是T類型 public <T> T test2(T t) { System.out.println(t); return t; } //該方法使用泛型T,E //參數(shù)包括T,E public <T, E> void test3(T t, E e) { System.out.println(t); System.out.println(e); } }
當(dāng)然這并不是泛型方法的全部,泛型方法可以出現(xiàn)雜任何地方和任何場景中使用。但是有一種情況是非常特殊的,當(dāng)泛型方法出現(xiàn)在泛型類中時,我們再通過一個例子看一下
//注意泛型類先寫類名再寫泛型,泛型方法先寫泛型再寫方法名 //類中聲明的泛型在成員和方法中可用 class A <T, E>{ { T t1 ; } A (T t){ this.t = t; } T t; public void test1() { System.out.println(this.t); } public void test2(T t,E e) { System.out.println(t); System.out.println(e); } } @Test public void run () { A <Integer,String > a = new A<>(1); a.test1(); a.test2(2,"ds"); // 1 // 2 // ds } static class B <T>{ T t; public void go () { System.out.println(t); } }
再看一個泛型方法和可變參數(shù)的例子:
public class 泛型和可變參數(shù) { @Test public void test () { printMsg("dasd",1,"dasd",2.0,false); print("dasdas","dasdas", "aa"); } //普通可變參數(shù)只能適配一種類型 public void print(String ... args) { for(String t : args){ System.out.println(t); } } //泛型的可變參數(shù)可以匹配所有類型的參數(shù)。。有點無敵 public <T> void printMsg( T... args){ for(T t : args){ System.out.println(t); } } //打印結(jié)果: //dasd //1 //dasd //2.0 //false }
靜態(tài)方法有一種情況需要注意一下,那就是在類中的靜態(tài)方法使用泛型:靜態(tài)方法無法訪問類上定義的泛型;如果靜態(tài)方法操作的引用數(shù)據(jù)類型不確定的時候,必須要將泛型定義在方法上。
即:如果靜態(tài)方法要使用泛型的話,必須將靜態(tài)方法也定義成泛型方法 。
public class StaticGenerator<T> { .... .... /** * 如果在類中定義使用泛型的靜態(tài)方法,需要添加額外的泛型聲明(將這個方法定義成泛型方法) * 即使靜態(tài)方法要使用泛型類中已經(jīng)聲明過的泛型也不可以。 * 如:public static void show(T t){..},此時編譯器會提示錯誤信息: "StaticGenerator cannot be refrenced from static context" */ public static <T> void show(T t){ } }
泛型方法能使方法獨立于類而產(chǎn)生變化,以下是一個基本的指導(dǎo)原則:
無論何時,如果你能做到,你就該盡量使用泛型方法。也就是說,如果使用泛型方法將整個類泛型化,那么就應(yīng)該使用泛型方法。另外對于一個static的方法而已,無法訪問泛型類型的參數(shù)。所以如果static方法要使用泛型能力,就必須使其成為泛型方法。
在使用泛型的時候,我們還可以為傳入的泛型類型實參進(jìn)行上下邊界的限制,如:類型實參只準(zhǔn)傳入某種類型的父類或某種類型的子類。
為泛型添加上邊界,即傳入的類型實參必須是指定類型的子類型
public class 泛型通配符與邊界 { public void showKeyValue(Generic<Number> obj){ System.out.println("key value is " + obj.getKey()); } @Test public void main() { Generic<Integer> gInteger = new Generic<Integer>(123); Generic<Number> gNumber = new Generic<Number>(456); showKeyValue(gNumber); //泛型中的子類也無法作為父類引用傳入 // showKeyValue(gInteger); } //直接使用?通配符可以接受任何類型作為泛型傳入 public void showKeyValueYeah(Generic<?> obj) { System.out.println(obj); } //只能傳入number的子類或者number public void showKeyValue1(Generic<? extends Number> obj){ System.out.println(obj); } //只能傳入Integer的父類或者Integer public void showKeyValue2(Generic<? super Integer> obj){ System.out.println(obj); } @Test public void testup () { //這一行代碼編譯器會提示錯誤,因為String類型并不是Number類型的子類 //showKeyValue1(generic1); 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); showKeyValue1(generic2); showKeyValue1(generic3); showKeyValue1(generic4); } @Test public void testdown () { Generic<String> generic1 = new Generic<String>("11111"); Generic<Integer> generic2 = new Generic<Integer>(2222); Generic<Number> generic3 = new Generic<Number>(2); // showKeyValue2(generic1);本行報錯,因為String并不是Integer的父類 showKeyValue2(generic2); showKeyValue2(generic3); } }
== 關(guān)于泛型數(shù)組要提一下 ==
看到了很多文章中都會提起泛型數(shù)組,經(jīng)過查看sun的說明文檔,在java中是”不能創(chuàng)建一個確切的泛型類型的數(shù)組”的。
也就是說下面的這個例子是不可以的: List<String>[] ls = new ArrayList<String>[10]; 而使用通配符創(chuàng)建泛型數(shù)組是可以的,如下面這個例子: 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泛型的擦除機(jī)制,在運行時JVM是不知道泛型信息的,所以可以給oa[1]賦上一個ArrayList而不會出現(xiàn)異常,但是在取出數(shù)據(jù)的時候卻要做一次類型轉(zhuǎn)換,所以就會出現(xiàn)ClassCastException,如果可以進(jìn)行泛型數(shù)組的聲明,上面說的這種情況在編譯期將不會出現(xiàn)任何的警告和錯誤,只有在運行時才會出錯。
而對泛型數(shù)組的聲明進(jìn)行限制,對于這樣的情況,可以在編譯期提示代碼有類型安全問題,比沒有任何提示要強(qiáng)很多。
下面采用通配符的方式是被允許的:數(shù)組的類型不可以是類型變量,除非是采用通配符的方式,因為對于通配符的方式,最后取出數(shù)據(jù)是要做顯式的類型轉(zhuǎn)換的。
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
最后
本文中的例子主要是為了闡述泛型中的一些思想而簡單舉出的,并不一定有著實際的可用性。另外,一提到泛型,相信大家用到最多的就是在集合中,其實,在實際的編程過程中,自己可以使用泛型去簡化開發(fā),且能很好的保證代碼質(zhì)量。
Java中的泛型是什么 ? 使用泛型的好處是什么?
這是在各種Java泛型面試中,一開場你就會被問到的問題中的一個,主要集中在初級和中級面試中。那些擁有Java1.4或更早版本的開發(fā)背景的人 都知道,在集合中存儲對象并在使用前進(jìn)行類型轉(zhuǎn)換是多么的不方便。泛型防止了那種情況的發(fā)生。它提供了編譯期的類型安全,確保你只能把正確類型的對象放入 集合中,避免了在運行時出現(xiàn)ClassCastException。
Java的泛型是如何工作的 ? 什么是類型擦除 ?
這是一道更好的泛型面試題。泛型是通過類型擦除來實現(xiàn)的,編譯器在編譯時擦除了所有類型相關(guān)的信息,所以在運行時不存在任何類型相關(guān)的信息。例如 List
什么是泛型中的限定通配符和非限定通配符 ?
這是另一個非常流行的Java泛型面試題。限定通配符對類型進(jìn)行了限制。有兩種限定通配符,一種是<? extends T>它通過確保類型必須是T的子類來設(shè)定類型的上界,另一種是<? super T>它通過確保類型必須是T的父類來設(shè)定類型的下界。泛型類型必須用限定內(nèi)的類型來進(jìn)行初始化,否則會導(dǎo)致編譯錯誤。另一方面<?>表 示了非限定通配符,因為<?>可以用任意類型來替代。更多信息請參閱我的文章泛型中限定通配符和非限定通配符之間的區(qū)別。
List<? extends T>和List <? super T>之間有什么區(qū)別 ?
這和上一個面試題有聯(lián)系,有時面試官會用這個問題來評估你對泛型的理解,而不是直接問你什么是限定通配符和非限定通配符。這兩個List的聲明都是 限定通配符的例子,List<? extends T>可以接受任何繼承自T的類型的List,而List<? super T>可以接受任何T的父類構(gòu)成的List。例如List<? extends Number>可以接受List
如何編寫一個泛型方法,讓它能接受泛型參數(shù)并返回泛型類型?
編寫泛型方法并不困難,你需要用泛型類型來替代原始類型,比如使用T, E or K,V等被廣泛認(rèn)可的類型占位符。泛型方法的例子請參閱Java集合類框架。最簡單的情況下,一個泛型方法可能會像這樣:
public V put(K key, V value) {
return cache.put(key, value);
}
Java中如何使用泛型編寫帶有參數(shù)的類?
這是上一道面試題的延伸。面試官可能會要求你用泛型編寫一個類型安全的類,而不是編寫一個泛型方法。關(guān)鍵仍然是使用泛型類型來代替原始類型,而且要使用JDK中采用的標(biāo)準(zhǔn)占位符。
編寫一段泛型程序來實現(xiàn)LRU緩存?
對于喜歡Java編程的人來說這相當(dāng)于是一次練習(xí)。給你個提示,LinkedHashMap可以用來實現(xiàn)固定大小的LRU緩存,當(dāng)LRU緩存已經(jīng)滿 了的時候,它會把最老的鍵值對移出緩存。LinkedHashMap提供了一個稱為removeEldestEntry()的方法,該方法會被put() 和putAll()調(diào)用來刪除最老的鍵值對。當(dāng)然,如果你已經(jīng)編寫了一個可運行的JUnit測試,你也可以隨意編寫你自己的實現(xiàn)代碼。
你可以把List
對任何一個不太熟悉泛型的人來說,這個Java泛型題目看起來令人疑惑,因為乍看起來String是一種Object,所以 List
“Java中的泛型怎么應(yīng)用”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。