[轉載] Java泛型詳解:<T>和Class<T>的使用。泛型類,泛型方法的詳細使用實例

參考鏈接: Java中的main()函數是強制性的嗎

一、引入?

?

1、泛型是什么?

首先告訴大家ArrayList就是泛型。那ArrayList能完成哪些想不到的功能呢?先看看下面這段代碼:

?

?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? ArrayList<String> strList = new ArrayList<String>();? ?ArrayList<Integer> intList = new ArrayList<Integer>();? ?ArrayList<Double> doubleList = new ArrayList<Double>();? ?

?

大家對ArrayList很熟悉,這里構造了三個List,分別盛裝String、Integer和Double;這就是ArrayList的過人之處:即各種類型的變量都可以組裝成對應的List,而不必針對每個類型分別實現一個構建ArrayList的類。這里可能看不懂,開篇總是困難的,下面看看如果沒有泛型的話,我們要怎么做;

?

2、沒有泛型會怎樣?

先看下面這段代碼:

?

我們實現兩個能夠設置點坐標的類,分別設置Integer類型的點坐標和Float類型的點坐標:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //設置Integer類型的點坐標? ?class IntegerPoint{? ? ? ?private Integer x ;? ? ? ?// 表示X坐標? ? ? ?private Integer y ;? ? ? ?// 表示Y坐標? ? ? ?public void setX(Integer x){? ? ? ? ? ?this.x = x ;? ? ? ?}? ? ? ?public void setY(Integer y){? ? ? ? ? ?this.y = y ;? ? ? ?}? ? ? ?public Integer getX(){? ? ? ? ? ?return this.x ;? ? ? ?}? ? ? ?public Integer getY(){? ? ? ? ? ?return this.y ;? ? ? ?}? ?}? ?//設置Float類型的點坐標? ?class FloatPoint{? ? ? ?private Float x ;? ? ? ?// 表示X坐標? ? ? ?private Float y ;? ? ? ?// 表示Y坐標? ? ? ?public void setX(Float x){? ? ? ? ? ?this.x = x ;? ? ? ?}? ? ? ?public void setY(Float y){? ? ? ? ? ?this.y = y ;? ? ? ?}? ? ? ?public Float getX(){? ? ? ? ? ?return this.x ;? ? ? ?}? ? ? ?public Float getY(){? ? ? ? ? ?return this.y ;? ? ? ?}? ?}? ?

?

那現在有個問題:大家有沒有發現,他們除了變量類型不一樣,一個是Integer一個是Float以外,其它并沒有什么區別!那我們能不能合并成一個呢?

?

答案是可以的,因為Integer和Float都是派生自Object的,我們用下面這段代碼代替:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class ObjectPoint{? ? ? ?private Object x ;? ? ? ?private Object y ;? ? ? ?public void setX(Object x){? ? ? ? ? ?this.x = x ;? ? ? ?}? ? ? ?public void setY(Object y){? ? ? ? ? ?this.y = y ;? ? ? ?}? ? ? ?public Object getX(){? ? ? ? ? ?return this.x ;? ? ? ?}? ? ? ?public Object getY(){? ? ? ? ? ?return this.y ;? ? ? ?}? ?}? ?

?

即全部都用Object來代替所有的子類;

?

在使用的時候是這樣的:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? ObjectPoint integerPoint = new ObjectPoint();? ?integerPoint.setX(new Integer(100));? ?Integer integerX=(Integer)integerPoint.getX();? ?

?

在設置的時候,使用new Integer(100)來新建一個Integer

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? integerPoint.setX(new Integer(100));? ?

?

然后在取值的時候,進行強制轉換:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? Integer integerX=(Integer)integerPoint.getX();? ?

?

由于我們設置的時候,是設置的Integer,所以在取值的時候,強制轉換是不會出錯的。

?

同理,FloatPoint的設置和取值也是類似的,代碼如下:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? ObjectPoint floatPoint = new ObjectPoint();? ?floatPoint.setX(new Float(100.12f));? ?Float floatX = (Float)floatPoint.getX();? ?

?

但問題來了:注意,注意,我們這里使用了強制轉換,我們這里setX()和getX()寫得很近,所以我們明確的知道我們傳進去的是Float類型,那如果我們記錯了呢?

?

比如我們改成下面這樣,編譯時會報錯嗎:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? ObjectPoint floatPoint = new ObjectPoint();? ?floatPoint.setX(new Float(100.12f));? ?String floatX = (String)floatPoint.getX();? ?

?

不會!!!我們問題的關鍵在于這句:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? String floatX = (String)floatPoint.getX();? ?

?

強制轉換時,會不會出錯。因為編譯器也不知道你傳進去的是什么,而floatPoint.getX()返回的類型是Object,所以編譯時,將Object強轉成String是成立的。必然不會報錯。

?

而在運行時,則不然,在運行時,floatPoint實例中明明傳進去的是Float類型的變量,非要把它強轉成String類型,肯定會報類型轉換錯誤的!

?

那有沒有一種辦法在編譯階段,即能合并成同一個,又能在編譯時檢查出來傳進去類型不對呢?當然,這就是泛型。

?

下面我們將對泛型的寫法和用法做一一講解。

?

二、各種泛型定義及使用?

?

?

1、泛型類定義及使用?

我們先看看泛型的類是怎么定義的:

?

?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //定義? ?class Point<T>{// 此處可以隨便寫標識符號? ? ? ? private T x ;? ? ? ? ? ? ?private T y ;? ? ? ? ? ? ?public void setX(T x){//作為參數? ? ? ? ? ?this.x = x ;? ? ? ?}? ? ? ?public void setY(T y){? ? ? ? ? ?this.y = y ;? ? ? ?}? ? ? ?public T getX(){//作為返回值? ? ? ? ? ?return this.x ;? ? ? ?}? ? ? ?public T getY(){? ? ? ? ? ?return this.y ;? ? ? ?}? ?};? ?//IntegerPoint使用? ?Point<Integer> p = new Point<Integer>() ;? ? p.setX(new Integer(100)) ;? ? System.out.println(p.getX());? ? ? ? //FloatPoint使用? ?Point<Float> p = new Point<Float>() ;? ? p.setX(new Float(100.12f)) ;? ? System.out.println(p.getX());? ? ?

?

先看看運行結果:

?

?

?

?

從結果中可以看到,我們實現了開篇中IntegerPoint類和FloatPoint類的效果。下面來看看泛型是怎么定義及使用的吧。??

(1)、定義泛型:Point<T> 首先,大家可以看到Point<T>,即在類名后面加一個尖括號,括號里是一個大寫字母。這里寫的是T,其實這個字母可以是任何大寫字母,大家這里先記著,可以是任何大寫字母,意義是相同的。 (2)類中使用泛型 這個T表示派生自Object類的任何類,比如String,Integer,Double等等。這里要注意的是,T一定是派生于Object類的。為方便起見,大家可以在這里把T當成String,即String在類中怎么用,那T在類中就可以怎么用!所以下面的:定義變量,作為返回值,作為參數傳入的定義就很容易理解了。??

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //定義變量? ?private T x ;? ? //作為返回值? ?public T getX(){? ? ? ? return x ;? ? ?}? ? ?//作為參數? ?public void setX(T x){? ? ? ? ?this.x = x ;? ? ?}? ??

?

(3)使用泛型類

?

下面是泛型類的用法:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //IntegerPoint使用? ?Point<Integer> p = new Point<Integer>() ;? ? p.setX(new Integer(100)) ;? ? System.out.println(p.getX());? ? ? ? //FloatPoint使用? ?Point<Float> p = new Point<Float>() ;? ? p.setX(new Float(100.12f)) ;? ? System.out.println(p.getX());? ? ?

?

首先,是構造一個實例:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? Point<String> p = new Point<String>() ;? ??

?

這里與普通構造類實例的不同之點在于,普通類構造函數是這樣的:Point p = new Point() ;?

?

而泛型類的構造則需要在類名后添加上<String>,即一對尖括號,中間寫上要傳入的類型。

?

因為我們構造時,是這樣的:class Point<T>,所以在使用的時候也要在Point后加上類型來定義T代表的意義。

?

然后在getVar()和setVar()時就沒有什么特殊的了,直接調用即可。

?

從上面的使用時,明顯可以看出泛型的作用,在構造泛型類的實例的時候:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //IntegerPoint使用? ?Point<Integer> p = new Point<Integer>() ;? ? //FloatPoint使用? ?Point<Float> p = new Point<Float>() ;? ??

?

尖括號中,你傳進去的是什么,T就代表什么類型。這就是泛型的最大作用,我們只需要考慮邏輯實現,就能拿給各種類來用。

?

前面我們提到ArrayList也是泛型,我們順便它的實現:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public class ArrayList<E>{? ? ? ?…………? ?}? ?

?

看到了吧,跟我們的Point實現是一樣的,這也就是為什么ArrayList能夠盛裝各種類型的主要原因。

?

(4)使用泛型實現的優勢

?

相比我們開篇時使用Object的方式,有兩個優點:

?

(1)、不用強制轉換

?

?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //使用Object作為返回值,要強制轉換成指定類型? ?Float floatX = (Float)floatPoint.getX();? ?//使用泛型時,不用強制轉換,直接出來就是String? ?System.out.println(p.getVar());? ??

?

(2)、在settVar()時如果傳入類型不對,編譯時會報錯

?

?

?

?

可以看到,當我們構造時使用的是String,而在setVar時,傳進去Integer類型時,就會報錯。而不是像Object實現方式一樣,在運行時才會報強制轉換錯誤。??

?

2、多泛型變量定義及字母規范?

(1)、多泛型變量定義

?

上在我們只定義了一個泛型變量T,那如果我們需要傳進去多個泛型要怎么辦呢?

?

只需要在類似下面這樣就可以了:

?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class MorePoint<T,U>{? ?}? ?

?

也就是在原來的T后面用逗號隔開,寫上其它的任意大寫字母即可。想加幾個就加幾個,比如我們想加五個泛型變量,那應該是這樣的:?

?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class MorePoint<T,U,A,B,C>{? ?}? ?

?

舉個粟子,我們在Point上再另加一個字段name,也用泛型來表示,那要怎么做?代碼如下:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class MorePoint<T,U> {? ? ? ?private T x;? ? ? ?private T y;? ? ? ? ? ? ? ? ?private U name;? ? ? ? ? public void setX(T x) {? ? ? ? ? ?this.x = x;? ? ? ?}? ? ? ?public T getX() {? ? ? ? ? ?return this.x;? ? ? ?}? ? ? ?…………? ? ? ?public void setName(U name){? ? ? ? ? ?this.name = name;? ? ? ?}? ? ? ? ? public U getName() {? ? ? ? ? ?return this.name;? ? ? ?}? ?}? ?//使用? ?MorePoint<Integer,String> morePoint = new MorePoint<Integer, String>();? ?morePoint.setName("harvic");? ?Log.d(TAG, "morPont.getName:" + morePoint.getName());? ?

?

從上面的代碼中,可以明顯看出,就是在新添加的泛型變量U用法與T是一樣的。

?

(2)、字母規范

?

在定義泛型類時,我們已經提到用于指定泛型的變量是一個大寫字母:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class Point<T>{? ? …………? ?}? ?

?

當然不是的!!!!任意一個大寫字母都可以。他們的意義是完全相同的,但為了提高可讀性,大家還是用有意義的字母比較好,一般來講,在不同的情境下使用的字母意義如下:

?

?E — Element,常用在java Collection里,如:List<E>,Iterator<E>,Set<E> K,V — Key,Value,代表Map的鍵值對 N — Number,數字 T — Type,類型,如String,Integer等等?

如果這些還不夠用,那就自己隨便取吧,反正26個英文字母呢。

?

再重復一遍,使用哪個字母是沒有特定意義的!只是為了提高可讀性!!!!

?

3、泛型接口定義及使用?

在接口上定義泛型與在類中定義泛型是一樣的,代碼如下:?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? interface Info<T>{? ? ? ? // 在接口上定義泛型? ? ? ? ?public T getVar() ; // 定義抽象方法,抽象方法的返回值就是泛型類型? ? ? ? ?public void setVar(T x);? ?}? ? ?

?

?

與泛型類的定義一樣,也是在接口名后加尖括號; (1)、使用方法一:非泛型類 但是在使用的時候,就出現問題了,我們先看看下面這個使用方法:??

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class InfoImpl implements Info<String>{? ?// 定義泛型接口的子類? ? ? ?private String var ;? ? ? ? ? ? ? ? // 定義屬性? ? ? ?public InfoImpl(String var){? ? ? ? // 通過構造方法設置屬性內容? ? ? ? ? ?this.setVar(var) ;? ? ? ?}? ? ? ?@Override? ? ? ?public void setVar(String var){? ? ? ? ? ?this.var = var ;? ? ? ?}? ? ? ?@Override? ? ? ?public String getVar(){? ? ? ? ? ?return this.var ;? ? ? ?}? ?}? ? ? public class GenericsDemo24{? ? ? ?public? void main(String arsg[]){? ? ? ? ? ?InfoImpl i = new InfoImpl("harvic");? ? ? ? ? ?System.out.println(i.getVar()) ;? ? ? ?}? ?};? ?

?

首先,先看InfoImpl的定義:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class InfoImpl implements Info<String>{? ? ? ?…………? ?}? ?

?

要清楚的一點是InfoImpl不是一個泛型類!因為他類名后沒有<T>!

?

然后在在這里我們將Info<String>中的泛型變量T定義填充為了String類型。所以在重寫時setVar()和getVar()時,IDE會也我們直接生成String類型的重寫函數。

?

最后在使用時,沒什么難度,傳進去String類型的字符串來構造InfoImpl實例,然后調用它的函數即可。

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public class GenericsDemo24{? ? ? ?public? void main(String arsg[]){? ? ? ? ? ?InfoImpl i = new InfoImpl("harvic");? ? ? ? ? ?System.out.println(i.getVar()) ;? ? ? ?}? ?};? ?

?

(2)、使用方法二:泛型類

?

在方法一中,我們在類中直接把Info<T>接口給填充好了,但我們的類,是可以構造成泛型類的,那我們利用泛型類來構造填充泛型接口會是怎樣呢??

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? interface Info<T>{? ? ? ? // 在接口上定義泛型? ? ? ?public T getVar() ; // 定義抽象方法,抽象方法的返回值就是泛型類型? ? ? ?public void setVar(T var);? ?}? ?class InfoImpl<T> implements Info<T>{? ?// 定義泛型接口的子類? ? ? ?private T var ;? ? ? ? ? ? ?// 定義屬性? ? ? ?public InfoImpl(T var){? ? ?// 通過構造方法設置屬性內容? ? ? ? ? ?this.setVar(var) ;? ? ? ? ?}? ? ? ?public void setVar(T var){? ? ? ? ? ?this.var = var ;? ? ? ?}? ? ? ?public T getVar(){? ? ? ? ? ?return this.var ;? ? ? ?}? ?}? ?public class GenericsDemo24{? ? ? ?public static void main(String arsg[]){? ? ? ? ? ?InfoImpl<String> i = new InfoImpl<String>("harvic");? ? ? ? ? ?System.out.println(i.getVar()) ;? ? ? ?}? ?};? ?

?

?

最關鍵的是構造泛型類的過程:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class InfoImpl<T> implements Info<T>{? ?// 定義泛型接口的子類? ? ? ?private T var ;? ? ? ? ? ? ?// 定義屬性? ? ? ?public InfoImpl(T var){? ? ?// 通過構造方法設置屬性內容? ? ? ? ? ?this.setVar(var) ;? ? ? ? ?}? ? ? ?public void setVar(T var){? ? ? ? ? ?this.var = var ;? ? ? ?}? ? ? ?public T getVar(){? ? ? ? ? ?return this.var ;? ? ? ?}? ?}? ?

?

在這個類中,我們構造了一個泛型類InfoImpl<T>,然后把泛型變量T傳給了Info<T>,這說明接口和泛型類使用的都是同一個泛型變量。

?

然后在使用時,就是構造一個泛型類的實例的過程,使用過程也不變。

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public class GenericsDemo24{? ? ? ?public static void main(String arsg[]){? ? ? ? ? ?Info<String> i = new InfoImpl<String>("harvic");? ? ? ? ? ?System.out.println(i.getVar()) ;? ? ? ?}? ?};? ?

?

使用泛型類來繼承泛型接口的作用就是讓用戶來定義接口所使用的變量類型,而不是像方法一那樣,在類中寫死。

?

那我們稍微加深點難度,構造一個多個泛型變量的類,并繼承自Info接口:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? class InfoImpl<T,K,U> implements Info<U>{? ?// 定義泛型接口的子類? ? ? ? private U var ;? ? ? ? ? ? private T x;? ? ? ? private K y;? ? ? ? public InfoImpl(U var){? ? ? ? // 通過構造方法設置屬性內容? ? ? ? ? ? this.setVar(var) ;? ? ? ? }? ? ? ? public void setVar(U var){? ? ? ? ? ? this.var = var ;? ? ? ? }? ? ? ? public U getVar(){? ? ? ? ? ? return this.var ;? ? ? ? }? ? }? ?

?

在這個例子中,我們在泛型類中定義三個泛型變量T,K,U并且把第三個泛型變量U用來填充接口Info。所以在這個例子中Info所使用的類型就是由U來決定的。

?

使用時是這樣的:泛型類的基本用法,不再多講,代碼如下:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public class GenericsDemo24{? ? ? ?public? void main(String arsg[]){? ? ? ? ? ?InfoImpl<Integer,Double,String> i = new InfoImpl<Integer,Double,String>("harvic");? ? ? ? ? ?System.out.println(i.getVar()) ;? ? ? ?}? ?}? ?

?

4、泛型函數定義及使用?

上面我們講解了類和接口的泛型使用,下面我們再說說,怎么單獨在一個函數里使用泛型。比如我們在新建一個普通的類StaticFans,然后在其中定義了兩個泛型函數:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public class StaticFans {? ? ? ?//靜態函數? ? ? ?public static? <T> void StaticMethod(T a){? ? ? ? ? ?Log.d("harvic","StaticMethod: "+a.toString());? ? ? ?}? ? ? ?//普通函數? ? ? ?public? <T> void OtherMethod(T a){? ? ? ? ? ?Log.d("harvic","OtherMethod: "+a.toString());? ? ? ?}? ?}? ?

?

上面分別是靜態泛型函數和常規泛型函數的定義方法,與以往方法的唯一不同點就是在返回值前加上<T>來表示泛型變量。其它沒什么區別。

?

使用方法如下:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //靜態方法? ?StaticFans.StaticMethod("adfdsa");//使用方法一? ?StaticFans.<String>StaticMethod("adfdsa");//使用方法二? ? ? //常規方法? ?StaticFans staticFans = new StaticFans();? ?staticFans.OtherMethod(new Integer(123));//使用方法一? ?staticFans.<Integer>OtherMethod(new Integer(123));//使用方法二? ?

?

結果如下:

?

?

?

?

首先,我們看靜態泛型函數的使用方法:??

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? StaticFans.StaticMethod("adfdsa");//使用方法一? ?StaticFans.<String>StaticMethod("adfdsa");//使用方法二? ?

?

從結果中我們可以看到,這兩種方法的結果是完全一樣的,但他們還有些區別的,區別如下:

?

方法一,可以像普通方法一樣,直接傳值,任何值都可以(但必須是派生自Object類的類型,比如String,Integer等),函數會在內部根據傳進去的參數來識別當前T的類別。但盡量不要使用這種隱式的傳遞方式,代碼不利于閱讀和維護。因為從外觀根本看不出來你調用的是一個泛型函數。

?

方法二,與方法一不同的地方在于,在調用方法前加了一個<String>來指定傳給<T>的值,如果加了這個<String>來指定參數的值的話,那StaticMethod()函數里所有用到的T類型也就是強制指定了是String類型。這是我們建議使用的方式。

?

同樣,常規泛型函數的使用也有這兩種方式:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? StaticFans staticFans = new StaticFans();? ?staticFans.OtherMethod(new Integer(123));//使用方法一? ?staticFans.<Integer>OtherMethod(new Integer(123));//使用方法二? ?

?

可以看到,與平常一樣,先創建類的實例,然后調用泛型函數。

?

方法一,隱式傳遞了T的類型,與上面一樣,不建議這么做。

?

方法二,顯示將T賦值為Integer類型,這樣OtherMethod(T a)傳遞過來的參數如果不是Integer那么編譯器就會報錯。

?

?

進階:返回值中存在泛型

?

上面我們的函數中,返回值都是void,但現實中不可能都是void,有時,我們需要將泛型變量返回,比如下面這個函數:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public static <T> List<T> parseArray(String response,Class<T> object){? ? ? ?List<T> modelList = JSON.parseArray(response, object);? ? ? ?return modelList;? ?}? ?

?

函數返回值是List<T>類型。至于傳入參數Class<T> object的意義,我們下面會講。這里也就是想通過這個例子來告訴大家,泛型變量其實跟String,Integer,Double等等的類的使用上沒有任何區別,T只是一個符號,可以代表String,Integer,Double……這些類的符號,在泛型函數使用時,直接把T看到String,Integer,Double……中的任一個來寫代碼就可以了。唯一不同的是,要在函數定義的中在返回值前加上<T>標識泛型;

?

5、其它用法:Class<T>類傳遞及泛型數組?

(1)、使用Class<T>傳遞泛型類Class對象

?

有時,我們會遇到一個情況,比如,我們在使用JSON解析字符串的時候,代碼一般是這樣的

?

?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public static List<SuccessModel> parseArray(String response){? ? ? ?List<SuccessModel> modelList = JSON.parseArray(response, SuccessModel.class);? ? ? ?return modelList;? ?}? ?

?

其中SuccessModel是自定義的解析類,代碼如下,其實大家不用管SuccessModel的定義,只考慮上面的那段代碼就行了。寫出來SuccessModel的代碼,只是不想大家感到迷惑,其實,這里只是fastJson的基本用法而已。

?

這段代碼的意義就是根據SuccessModel解析出List<SuccessModel>的數組。

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public class SuccessModel {? ? ? ?private boolean success;? ? ? ? ? ? ? public boolean isSuccess() {? ? ? ? ? ?return success;? ? ? ?}? ? ? ? ? public void setSuccess(boolean success) {? ? ? ? ? ?this.success = success;? ? ? ?}? ?}? ??

?

那現在,我們把下面這句組裝成一個泛型函數要怎么來做呢?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public static List<SuccessModel> parseArray(String response){? ? ? ?List<SuccessModel> modelList = JSON.parseArray(response, SuccessModel.class);? ? ? ?return modelList;? ?}? ?

?

首先,我們應該把SuccessModel單獨抽出來做為泛型變量,但parseArray()中用到的SuccessModel.class要怎么弄呢?

?

先來看代碼:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public static <T> List<T> parseArray(String response,Class<T> object){? ? ? ?List<T> modelList = JSON.parseArray(response, object);? ? ? ?return modelList;? ?}? ?

?

注意到,我們用的Class<T> object來傳遞類的class對象,即我們上面提到的SuccessModel.class。

?

這是因為Class<T>也是一泛型,它是傳來用來裝載類的class對象的,它的定義如下:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public final class Class<T> implements Serializable {? ? ? ?…………? ?}? ?

?

通過Class<T>來加載泛型的Class對象的問題就講完了,下面來看看泛型數組的使用方法吧。

?

(2)、定義泛型數組

?

在寫程序時,大家可能會遇到類似String[] list = new String[8];的需求,這里可以定義String數組,當然我們也可以定義泛型數組,泛型數組的定義方法為 T[],與String[]是一致的,下面看看用法:?

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? //定義? ?public static <T> T[] fun1(T...arg){? // 接收可變參數? ? ? ? ? ? return arg ;? ? ? ? ? ? // 返回泛型數組? ? ?}? ? ?//使用? ?public static void main(String args[]){? ? ? ? ? ? Integer i[] = fun1(1,2,3,4,5,6) ;? ? ? ? ? Integer[] result = fun1(i) ;? ?}? ? ?

?

我們先看看 定義時的代碼:

?

?

??

? ?

? ?[java]?

? ?view plain

? ? copy?

? ??

? ??

? ?

? ?

??

? public static <T> T[] fun1(T...arg){? // 接收可變參數? ? ? ? ? ? return arg ;? ? ? ? ? ? // 返回泛型數組? ? ?}? ? ?

?

首先,定義了一個靜態函數,然后定義返回值為T[],參數為接收的T類型的可變長參數。如果有同學對T...arg的用法不了解,可以去找下JAVA 可變長參數方面的知識。

?

由于可變長參數在輸入后,會保存在arg這個數組中,所以,我們直接把數組返回即可。

?

??

轉自:http://blog.csdn.net/harvic880925/article/details/49872903?

??

?

下面是我自己實際使用泛型的幾個實例。 希望看完上面的文章之后,再看實際使用的例子,可以更好的理解和使用和實際使用。

?

?

??

關于泛型類的使用實例?

?

import lombok.Data;

?

@Data

public class MultiObject<T> {

?

? ? /**

? ? ?* 成功狀態

? ? ?*/

? ? private boolean success;

?

? ? /**

? ? ?* 異常

? ? ?*/

? ? private Exception ex;

?

? ? /**

? ? ?* 數據

? ? ?*/

? ? private T obj;

? ??

? ? public MultiObject() {

? ? }

?

? ? /**

? ? ?* 注意:當傳入的泛型是Boolean時,就和第三個構造函數沖突了。

? ? ?*/

? ? public MultiObject(boolean success) {

? ? ? ? this.success = success;

? ? }

? ??

? ? public MultiObject(Exception ex) {

? ? ? ? this.success = false;

? ? ? ? this.ex = ex;

? ? }

? ??

? ? public MultiObject(T value) {

? ? ? ? this.success = true;

? ? ? ? this.obj = value;

? ? }

}

?

簡單解釋下這個model。

?在實際業務代碼里面,可能有很多種操作,然后我們關心這個操作的執行結果,主要有幾點。

?1,成功與否。對應屬性success。

?2,異常信息。對應屬性ex。若是操作正常執行,則就不在意這個屬性的值。

?3,我們操作的最終目的對象。對應屬性obj。?

?

?泛型牛逼的地方就是在這個地方。 如果你不用泛型,而使用Object類型,那么每次執行完之后,我們即使得到這個結果,還得類型轉換一下,那么這下就像文章上面描述的那樣。分分鐘出現castfailexception。也就是類型轉換異常啦。 但是,若是使用了這個泛型之后,那么我們的某個操作所需要的返回結果是什么類型,就可以 傳入什么類型,而且在實際取得返回結果的時候,就不需要使用類型轉換,這樣就很好的達到了目的。 這個主要是代碼設計層次的提高。寫再多的業務代碼,要是不提高,那么寫的都是渣。? 關于,這個model代碼里面為啥沒有getter和setter,都是因為使用@Data這個注解,可以自動填充這個getter和setter。所以。就表在意這個問題啦。在其他地方可以正常使用各個屬性getter和setter方法,雖然這些方法,你暫時看不見。有興趣的可以了解下lombok。??

??

?

關于泛型方法的使用實例 這個地方就有2個,但是上面文章也都講到啦。

?1,一個是泛型表示某一個類型的參數。為的傳遞某一類的參數對象

?2,另一個則是傳遞的不是參數,而是代表Class,某一個類。

?

?

?

恰巧我都使用過,就正好記錄一下實際使用實例。?

?

? ? /**

? ? ?* 將Json字符串信息轉換成對應的Java對象

? ? ?*

? ? ?* @param json json字符串對象

? ? ?* @param c? ? 對應的類型

? ? ?*/

? ? public static <T> T parseJsonToObj(String json, Class<T> c) {

? ? ? ? try {

? ? ? ? ? ? JSONObject jsonObject = JSONObject.parseObject(json);

? ? ? ? ? ? return JSON.toJavaObject(jsonObject, c);

? ? ? ? } catch (Exception e) {

? ? ? ? ? ? LOG.error(e.getMessage());

? ? ? ? }

? ? ? ? return null;

? ? }然后是具體調用的地方的代碼。?

?

?

Collector collectorObj = JSONUtils.parseJsonToObj(collector, Collector.class);

Flume flume = JSONUtils.parseJsonToObj(flumeJson, Flume.class);

Probe probe = JSONUtils.parseJsonToObj(probeJson, Probe.class);可以看到,真的只是因為傳入的參數類型不一樣,但若你不知道泛型的話,那你就得沒遇到一個類型的轉換,你就得寫一個這么個方法。?

??

?

? ? /**

? ? ?* @param dest? ?目的集合

? ? ?* @param source 源集合

? ? ?* @param <T>? ? 集合參數的類型

? ? ?*/

? ? private static <T> void listAddAllAvoidNPE(List<T> dest, List<T> source) {

? ? ? ? if (source == null) {

? ? ? ? ? ? return;

? ? ? ? }

? ? ? ? dest.addAll(source);

? ? }

?

? ? private static <T> void listAddAvoidNull(List<T> dest, T source) {

? ? ? ? if (source == null) {

? ? ? ? ? ? return;

? ? ? ? }

? ? ? ? dest.add(source);

? ? }

?這個就是傳入的參數為某一類的參數,主要是要使用參數對象,而不是上面的那個使用的參數的類Class?

?我這方法提出來,主要是因為,直接使用list類的addAll()方法,如果添加的是null,那么就會拋異常。但是總不能我在所有使用的地方,都判斷一下我要添加的參數是不是null,然后再調用list的addAll()方法吧。那樣的話,這樣的判斷代碼會啰嗦的海了去了。所以,就這么提出來了。?

?

這個時候,這個T,使用起來就像使用我們常用的一般對象一樣,我這的參數是個List類型,當然也可是其他類型的,姿勢都一樣。?

然后是具體調用的地方的代碼?

?

? ? List<ProbeObject> list = Lists.newArrayList();

? ? listAddAllAvoidNPE(list, decoder.getProperties());這個方法的第二個參數的返回值可能是null,所以,直接調用addAll(),就會拋空指針異常。所以,就如上,那么一提取。就好多啦。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/540419.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/540419.shtml
英文地址,請注明出處:http://en.pswp.cn/news/540419.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

數字和數字根的總和_使用8086微處理器查找8位數字的數字總和

數字和數字根的總和Problem statement: 問題陳述&#xff1a; Write an assembly language program in 8086 microprocessor to find sum of digit of an 8 bits number using 8 bits operation. 在8086微處理器中編寫匯編語言程序&#xff0c;以使用8位運算找到8位數字的位數…

[轉載] Java筆試題集錦

參考鏈接&#xff1a; 關于Java中文件名和類名的誤解 Java筆試題集錦 1.MVC的各個部分都有那些技術來實現?如何實現? 答&#xff1a;MVC是Model&#xff0d;View&#xff0d;Controller的簡寫。"Model" 代表的是應用的業務邏輯&#xff08;通過JavaBean&#xff…

gcc -pthread_錯誤-在GCC Linux中使用C程序未定義對'pthread_create'的引用

gcc -pthread在Linux中修復對pthread_create的未定義引用 (Fixing undefined reference to pthread_create in Linux) This is a common error while compiling C program in GCC/G Linux. This error occurs when you are using pthread_create function to create threads in…

[轉載] Java面試題全集(上)

參考鏈接&#xff1a; 如何運行不同目錄中的Java類文件 2013年年底的時候&#xff0c;我看到了網上流傳的一個叫做《Java面試題大全》的東西&#xff0c;認真的閱讀了以后發現里面的很多題目是重復且沒有價值的題目&#xff0c;還有不少的參考答案也是錯誤的&#xff0c;于是我…

python重載運算符乘法_Python | 使用乘法運算符創建一個字符串的多個副本

python重載運算符乘法Given a string and we have to create its multiple copies by using multiplication operator in Python? 給定一個字符串&#xff0c;我們必須通過在Python中使用乘法運算符來創建其多個副本&#xff1f; If you want to create multiple copies of …

一次前端筆試總結

1.有一個長度未知的數組a&#xff0c;如果它的長度為0就把數字1添加到數組里面&#xff0c;否則按照先進先出的隊列規則讓第一個元素出隊。 分析&#xff1a;這道題主要是考核了數組的隊列方法和棧方法。另外&#xff0c;原題還有字數限制的&#xff0c;只有在字數小于30并且結…

Java文件類boolean setLastModified(long set_new_time)方法,包含示例

文件類boolean setLastModified(long set_new_time) (File Class boolean setLastModified(long set_new_time)) This method is available in package java.io.File.setLastModified(long set_new_time). 軟件包java.io.File.setLastModified(long set_new_time)中提供了此方法…

[轉載] Linux里面的文件目錄類指令

參考鏈接&#xff1a; 如何運行不同目錄中的Java類文件 引用&#xff1a;尚硅谷韓老師的《尚硅谷-Linux-經典升級》 日常總結 pwd 指令 &#xff08;顯示當前工作目錄的絕對路徑&#xff09; 基本語法 pwd (功能描述&#xff1a;顯示當前工作目錄的絕對路徑) …

[轉載] 微服務安全和治理

參考鏈接&#xff1a; 微服務介紹 在整體式架構中&#xff0c;由于運行應用程序的運行時環境相對隔離&#xff0c;所以治理和安全保護很簡單。微服務架構具有典型的革新特征&#xff0c;給活動的治理和應用程序的安全威脅保護帶來了更多挑戰。 微服務架構中的安全性 微服務…

SSL

今天遇到一位網友要求老蔣將他當前已經在使用的WDCP面板環境&#xff0c;給某個站點添加SSL證書&#xff0c;實現HTTPS網址訪問。在過去的幾篇文章中&#xff0c;老蔣也有分享過不少在Linux VPS中對應的WEB環境安裝SSL證書的經歷&#xff0c;其實總體來看都大同小異&#xff0c…

[轉載] Java中如何引用另一個類里的集合_Java工程師面試題整理

參考鏈接&#xff1a; 在Java中將預定義的類名用作類或變量名 花了一星期把學過的都整理一遍 盡量易懂&#xff0c;從基礎到框架 最新版大廠面經匯總出爐&#xff0c;持續更新中 匯總完了上傳網盤&#xff0c;設計到后端架構師的一切知識 如果沒更新就代表我死了 一&#xff0…

應用寶認領應用

2019獨角獸企業重金招聘Python工程師標準>>> 【Android】命令行jarsigner簽字和解決找不到證書鏈錯誤 1、簽名失敗 $jarsigner -verbose -keystore /Volumes/Study/resourcesLib/Qunero-achivements/AndroidApp/QuLordy-signed-key -signedjar ./signed_XiaomiVerif…

[轉載] Java | Java 面向對象知識小抄

參考鏈接&#xff1a; 在Java中將預定義的類名用作類或變量名 0. 前言 下面是本篇的內容提綱&#xff1a; 1. 類 Java 中類的聲明形式如下所示&#xff0c;變量的聲明和方法的定義意味著只能聲明變量、初始化、方法定義等&#xff0c;而不能在方法外進行賦值等操作。 …

bash顏色、變量、數組、相關腳本示例

下面是bash的相關內容&#xff0c;包括bash的顏色代碼、bash的四類文件、bash中變量處理方式、數組變量、shell的過程式編程語言以及部分簡單腳本例子。 一、bash的顏色顯示規則&#xff08;顏色代碼&#xff09; bash的顏色代碼&#xff0c;是ASCII編碼對于顏色進行設置。顏色…

[轉載] java程序員快速學c++

參考鏈接&#xff1a; 如何成為一名優秀的Java程序員 java程序員快速學c java程序員學cplus&#xff0c;其實大部分的語法是一樣的&#xff0c;但是對java程序員來將&#xff0c;還是有寫難點需要注意。看完這些還不能保證會寫c程序&#xff0c;不過一般的程序可以看懂&…

[轉載] Java標識符 數據類型 常量與變量

參考鏈接&#xff1a; Java標識符 Java標識符 Java對包、類、接口、方法、變量、常量&#xff08;不包括項目名&#xff09;等命名時使用的字符序列稱為標識符。 命名規范&#xff1a; 1.標識符由數字、字母、_、$組成&#xff0c;且首字母不能是數字。 2.標識符對大小…

設計模式單例

單例模式 設計原則&#xff1a;無常用場景&#xff1a;應用中有對象需要是全局的且唯一使用概率&#xff1a;99.99999%復雜度&#xff1a;低變化點&#xff1a;無選擇關鍵點&#xff1a;一個對象在應用中出現多個實例是否會引起邏輯上或者是程序上的錯誤逆鱗&#xff1a;在以為…

Squid服務日志分析

Squid服務日志分析 Apache 和 Squid 是兩種著名的代理緩存軟件&#xff0c;但Squid 較 Apache 而言是專門的代理緩存服務器軟件&#xff0c;其代理緩存的功能強大&#xff0c;支持 HTTP/1.1 協議&#xff0c;其緩存對象也較多&#xff1b;并且 Squid 的緩存管理模塊和訪問控制模…

[轉載] Java中的元數據

參考鏈接&#xff1a; Java中的數據類型 元數據 也可能剛聽到元數據你會有點陌生&#xff0c;其實任何一個使用過struts&#xff0c;ejb或者hibernate的開發人員都在不知不覺中使用元數據。所謂的元數據是指用來描述數據的數據&#xff0c;更通俗一點就是描述代碼間關系&#…

云時代 揭開性能監測戰略的隱秘優勢

云時代的性能監測戰略 能夠對各種變化做出快速響應而不偏離重心和企業發展動力&#xff0c;正逐漸成為各行各業、各種規模企業的奮斗目標。業務敏捷性通常是運營良好&#xff0c;可實現盈利的企業標志。實現這一目標意味著公司已經成功地利用業務關鍵型技術來提高生產率&#x…