日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 人文社科 > 生活经验 >内容正文

生活经验

java 泛型 .net_Java基础11:Java泛型详解

發布時間:2023/11/27 生活经验 39 豆豆
生活随笔 收集整理的這篇文章主要介紹了 java 泛型 .net_Java基础11:Java泛型详解 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

本文對java的泛型的概念和使用做了詳盡的介紹。

本文參考https://blog.csdn.net/s10461/article/details/53941091

具體代碼在我的GitHub中可以找到

文章首發于我的個人博客:

更多關于Java后端學習的內容請到我的CSDN博客上查看:

泛型概述

泛型在java中有很重要的地位,在面向對象編程及各種設計模式中有非常廣泛的應用。

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

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

一個栗子

一個被舉了無數次的例子: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可以存放任意類型,例子中添加了一個String類型,添加了一個Integer類型,再使用時都以String的方式使用,因此程序崩潰了。為了解決類似這樣的問題(在編譯階段就可以解決),泛型應運而生。

我們將第一行聲明初始化list的代碼更改一下,編譯器會在編譯階段就能夠幫我們發現類似這樣的問題。

List arrayList = new ArrayList(); ... //arrayList.add(100); 在編譯階段,編譯器就會報錯

特性

泛型只在編譯階段有效。看下面的代碼:List stringArrayList = new ArrayList();

List integerArrayList = new ArrayList();

Class classStringArrayList = stringArrayList.getClass();

Class classIntegerArrayList = integerArrayList.getClass();

if(classStringArrayList.equals(classIntegerArrayList)){

Log.d("泛型測試","類型相同");

}通過上面的例子可以證明,在編譯之后程序會采取去泛型化的措施。也就是說Java中的泛型,只在編譯階段有效。在編譯過程中,正確檢驗泛型結果后,會將泛型的相關信息擦出,并且在對象進入和離開方法的邊界處添加類型檢查和類型轉換的方法。也就是說,泛型信息不會進入到運行時階段。

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

泛型有三種使用方式,分別為:泛型類、泛型接口、泛型方法

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

泛型類的最基本寫法(這么看可能會有點暈,會在下面的例子中詳解):class 類名稱 {

private 泛型標識 /*(成員變量類型)*/ var;

.....

}

一個最普通的泛型類:

//此處T可以隨便寫為任意標識,常見的如T、E、K、V等形式的參數常用于表示泛型//在實例化泛型類時,必須指定T的具體類型

public class Generic{

//在類中聲明的泛型整個類里面都可以用,除了靜態部分,因為泛型是實例化時聲明的。

//靜態區域的代碼在編譯時就已經確定,只與類相關

class A {

T t;

}

//類里面的方法或類中再次聲明同名泛型是允許的,并且該泛型會覆蓋掉父類的同名泛型T

class B {

T t;

}

//靜態內部類也可以使用泛型,實例化時賦予泛型實際類型

static class C {

T t;

}

public static void main(String[] args) {

//報錯,不能使用T泛型,因為泛型T屬于實例不屬于類

// T t = null;

}

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

private T key;

public Generic(T key) { //泛型構造方法形參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 12345612-27 09:20:04.432 13063-13063/? D/泛型測試: key is key_vlaue定義的泛型類,就一定要傳入泛型類型實參么?并不是這樣,在使用泛型的時候如果傳入泛型實參,則會根據傳入的泛型實參做相應的限制,此時泛型才會起到本應起到的限制作用。如果不傳入泛型類型實參的話,在泛型類中使用泛型的方法或成員變量定義的類型可以為任何的類型。

看一個例子: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

注意: 泛型的類型參數只能是類類型,不能是簡單類型。 不能對確切的泛型類型使用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類中的功能。那么可以用 ? 通配符來表未知類型

public void showKeyValue(Generic obj){ System.out.println(obj); }Generic gInteger = new Generic(123);

Generic gNumber = new Generic(456);

public void test () {

// showKeyValue(gInteger);該方法會報錯

showKeyValue1(gInteger);

}

public void showKeyValue1(Generic> obj) {

System.out.println(obj);

}

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

// cannot be applied to Generic

// showKeyValue(gInteger);

泛型方法

在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"));

泛型方法的基本用法

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

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

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

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

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

* 如:public K showKeyName(Generic container){

* ...

* }

*/

public class 泛型方法 {

@Test

public void test() {

test1();

test2(new Integer(2));

test3(new int[3],new Object());

//打印結果

// null

// 2

// [I@3d8c7aca

// java.lang.Object@5ebec15

}

//該方法使用泛型T

public void test1() {

T t = null;

System.out.println(t);

}

//該方法使用泛型T

//并且參數和返回值都是T類型

public T test2(T t) {

System.out.println(t);

return t;

}

//該方法使用泛型T,E

//參數包括T,E

public void test3(T t, E e) {

System.out.println(t);

System.out.println(e);

}

}

類中的泛型方法

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

//類中聲明的泛型在成員和方法中可用

class A {

{

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 a = new A<>(1);

a.test1();

a.test2(2,"ds");

// 1

// 2

// ds

}

static class B {

T t;

public void go () {

System.out.println(t);

}

}

泛型方法與可變參數

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

@Test

public void test () {

printMsg("dasd",1,"dasd",2.0,false);

print("dasdas","dasdas", "aa");

}

//普通可變參數只能適配一種類型

public void print(String ... args) {

for(String t : args){

System.out.println(t);

}

}

//泛型的可變參數可以匹配所有類型的參數。。有點無敵

public void printMsg( T... args){

for(T t : args){

System.out.println(t);

}

}

//打印結果:

//dasd

//1

//dasd

//2.0

//false

}

靜態方法與泛型

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

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

....

....

/**

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

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

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

"StaticGenerator cannot be refrenced from static context"

*/

public static void show(T t){

}

}

泛型方法總結

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

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

泛型上下邊界

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

為泛型添加上邊界,即傳入的類型實參必須是指定類型的子類型public class 泛型通配符與邊界 {

public void showKeyValue(Generic obj){

System.out.println("key value is " + obj.getKey());

}

@Test

public void main() {

Generic gInteger = new Generic(123);

Generic gNumber = new Generic(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 generic1 = new Generic("11111");

Generic generic2 = new Generic(2222);

Generic generic3 = new Generic(2.4f);

Generic generic4 = new Generic(2.56);

showKeyValue1(generic2);

showKeyValue1(generic3);

showKeyValue1(generic4);

}

@Test

public void testdown () {

Generic generic1 = new Generic("11111");

Generic generic2 = new Generic(2222);

Generic generic3 = new Generic(2);

// showKeyValue2(generic1);本行報錯,因為String并不是Integer的父類

showKeyValue2(generic2);

showKeyValue2(generic3);

}

}

== 關于泛型數組要提一下 ==

看到了很多文章中都會提起泛型數組,經過查看sun的說明文檔,在java中是”不能創建一個確切的泛型類型的數組”的。也就是說下面的這個例子是不可以的:

List[] ls = new ArrayList[10];

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

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

這樣也是可以的:

List[] ls = new ArrayList[10];

下面使用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

最后

本文中的例子主要是為了闡述泛型中的一些思想而簡單舉出的,并不一定有著實際的可用性。另外,一提到泛型,相信大家用到最多的就是在集合中,其實,在實際的編程過程中,自己可以使用泛型去簡化開發,且能很好的保證代碼質量。

總結

以上是生活随笔為你收集整理的java 泛型 .net_Java基础11:Java泛型详解的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。