色综合老司机第九色激情 _中文字幕日韩av资源站_国产+人+亚洲_久久久精品影院_久久久视频免费观看_欧美激情亚洲自拍_亚洲成av人片在线观看香蕉_热草久综合在线_欧美极品第一页_2020国产精品自拍

千鋒教育-做有情懷、有良心、有品質的職業教育機構

手機站
千鋒教育

千鋒學習站 | 隨時隨地免費學

千鋒教育

掃一掃進入千鋒手機站

領取全套視頻
千鋒教育

關注千鋒學習站小程序
隨時隨地免費學習課程

當前位置:首頁  >  技術干貨  > Java之泛型基本使用

Java之泛型基本使用

來源:千鋒教育
發布人:zyh
時間: 2023-06-12 15:14:00 1686554040

  全文大約【4600】字,不說廢話,只講可以讓你學到技術、明白原理的純干貨!本文帶有豐富的案例及配圖視頻,讓你更好地理解和運用文中的技術概念,并可以給你帶來具有足夠啟迪的思考......

Java之泛型基本使用

  一. 泛型方法

  1.簡介

  我們可以在定義接口和類時使用泛型,這樣該接口和類中的所有方法及成員變量等處,也都可以使用該泛型。但其實泛型可以應用在整個類上,也可以只應用在類中的某個方法上。也就是說,方法所在的類可以是泛型類,也可以不是泛型類。方法中是否帶有泛型,與其所在的類有沒有泛型沒有關系。

  泛型方法是在調用方法時才確定類型的方法,泛型可以使得該方法獨立于類而產生變化。另外,static靜態方法無法訪問泛型類的類型參數,因此,如果想讓一個static方法具有泛型能力,就必須使該靜態方法成為泛型方法。

  2.語法

  我們在定義泛型方法時,需要在方法名的前面添加類型參數。定義泛型方法的語法格式如下: 

[訪問權限修飾符] [static] [final] <類型參數列表> 返回值類型 方法名([形式參數列表])

  例如:  

public static <T> List showInfo(Class<T> clazz,int userId){}

  一般情況下,我們編寫泛型方法時,必須在該方法的名稱前聲明要使用的泛型,并且可以同時聲明多個泛型,中間也是用逗號分割。接下來小編就定義一個泛型方法,給大家具體介紹一下泛型方法的創建和使用。

  3. 代碼案例

  這里我們定義一個泛型方法,用于對數組排序后再遍歷元素輸出,代碼如下:  

import java.util.Arrays;

/**
* @author 一一哥Sun
*/
public class Demo04 {

//定義了一個靜態的泛型方法,遍歷數組中的每個元素
public static <T> void printArray(T[] arr) {
//先對數組進行排序
Arrays.sort(arr);
//再遍歷數組元素
for (T t : arr) {
System.out.print(t + " ");
}
System.out.println();
}

public static void main(String[] args) {
Integer[] nums= {100,39,8,200,65};
//調用泛型方法
printArray(nums);
}
}

  在上面的代碼中,printArray()就是一個泛型方法,該方法中使用了類型參數T。并且我們在方法的參數中,使用類型參數T定義了一個泛型數組T[],接著對該數組進行排序和遍歷。這樣以后無論我們傳入任何類型的數組,都可以在不進行類型轉換的前提下,輕松實現排序等功能了,這樣我們之前提的需求也就很容易實現了。

  二. 通配符

  除了以上這些用法之外,泛型中還有一個很重要的通配符功能,接下來我們就來看看它是怎么回事。

  1.簡介

  泛型中的通配符其實也是一種特殊的泛型類型,也稱為通配符類型參數。利用通配符類型參數,可以讓我們編寫出更通用的代碼,甚至可以在不知道實際類型的情況下使用它們。我們一般是使用 ? 來代替具體的類型參數,例如 List 在邏輯上可以等同于 List、List 等所有 List<具體類型實參> 的類。

  對此,有的小伙伴可能會很好奇,我們為什么需要通配符呢?其實之所以會出現通配符,主要是在開發時,有時候我們需要一個泛型類型,但我們卻不知道該使用哪個具體的類型。在這種情況下,我們就可以使用通配符類型參數,讓代碼更加地通用。比如,我們想編寫一個可以接受任何類型的集合,并返回其中最大的元素時,此時我們可能并不確定到底該傳入哪個具體的集合,那使用通配符就會更好一些。

  2.通配符的形式

  泛型通配符在具體使用時,有如下三種實現形式:

  ● 未限定通配符(?):?表示未知類型的通配符;

  ● 上限通配符(? extends T):?表示類型上限的通配符,T是一個類或接口;

  ● 下限通配符(? super T):?表示類型下限的通配符,T是一個類或接口。

  接下來小編針對以上這三種形式,分別通過幾個案例來給大家講解其用法。

  3.未限定通配符(?)

  未限定通配符(?)是一種表示未知類型的通配符,它可以在需要一個類型參數的情況下使用。但由于沒有限制,因此它只能用于簡單的情況,例如集合中的迭代器或者返回類型是泛型的方法等。下面是一個簡單的例子:  

import java.util.ArrayList;
import java.util.List;

/**
* @author 一一哥Sun
*/
public class Demo05 {

public static void main(String[] args) {
List<String> names = new ArrayList<String>();
List<Integer> ages = new ArrayList<Integer>();
List<Number> numbers = new ArrayList<Number>();

names.add("一一哥");
names.add("秦始皇");
ages.add(28);
ages.add(50);
ages.add(28);
numbers.add(100);
numbers.add(800);

printElement(names);
printElement(ages);
printElement(numbers);
}

//未限定通配符的使用
public static void printElement(List<?> data) {
for(int i=0;i<data.size();i++) {
//data.getClass().getSimpleName():用于獲取某個類的類名
System.out.println(data.getClass().getSimpleName()+"--data: " + data.get(i));
}
}

}

  在這個例子中,printElement()方法就接受了一個未知類型的List集合,所以names,ages,numbers都可以作為這個方法的實參,這就是未限定通配符的作用。

  4. PECS原則

  PECS是Producer Extends Consumer Super的縮寫,這是關于Java泛型的一種設計原則。該原則表示,如果我們需要返回T,它是生產者(Producer),要使用extends通配符;如果需要寫入T,它就是消費者(Consumer),要使用super通配符。該原則可以指導我們在使用泛型時,該如何定義類型參數的上限和下限。

  當我們使用泛型時,可能需要定義類型參數的上限和下限。例如,我們想要編寫一個方法來處理一些集合類型,但我們不知道這些集合中到底有什么類型的元素,此時我們就可以定義一個類型參數來處理所有的集合類型。一般我們可以利用extends來設置泛型上限,利用super來設置泛型下限。接下來小編會在下面的第5和第6小結中,給大家講解泛型的上限和下限具體該如何實現,請大家繼續往下學習。

  5. 上限通配符(? extends T)

  上限通配符(?extends T)是一種表示類型上限的通配符,其中T是一個類或接口,泛型類的類型必須實現或繼承 T這個接口或類。它指定了可以使用的類型上限,主要是用于限制輸入的參數類型。 

import java.util.ArrayList;
import java.util.List;

/**
* @author 一一哥Sun
*/
public class Demo06 {

public static void main(String[] args) {
List<String> names = new ArrayList<String>();
List<Integer> ages = new ArrayList<Integer>();
List<Number> numbers = new ArrayList<Number>();

names.add("一一哥");
names.add("秦始皇");
ages.add(28);
ages.add(50);
ages.add(28);
numbers.add(100);
numbers.add(800);

//String等非Number類型就不行
//printElementUpbound(names);

//泛型值只能是Number及其子類類型,所以Integer/Double/Float等類型都可以,但String就不行
printElementUpbound(ages);
printElementUpbound(numbers);
}

//上限通配符的使用,這里的泛型值只能是Number及其子類類型
public static void printElementUpbound(List<? extends Number> data) {
for(int i=0;i<data.size();i++) {
//data.getClass().getSimpleName():用于獲取某個類的類名
System.out.println(data.getClass().getSimpleName()+"--data: " + data.get(i));
}
}
}

  在這個例子中,printElementUpbound方法中的集合泛型,可以是Number類或其子類,除此之外的其他類型都不行。也就是說,我們只能使用Number或其子類作為類型參數,泛型類型的上限是Number,這就是上限通配符的含義。

  6. 下限通配符(? super T)

  下限通配符(?super T)是一種表示類型下限的通配符,其中T是一個類或接口。它指定了可以使用的類型下限,主要用于限制輸出的參數類型。下面是一個簡單的例子: 

import java.util.ArrayList;
import java.util.List;

/**
* @author 一一哥Sun
* V我領資料:syc_2312119590
* 各平臺都有小編的同名博客哦
*/
public class Demo07 {

public static void main(String[] args) {
List<String> names = new ArrayList<String>();
List<Integer> ages = new ArrayList<Integer>();
List<Double> numbers = new ArrayList<Double>();

names.add("一一哥");
names.add("秦始皇");
ages.add(28);
ages.add(50);
ages.add(28);
numbers.add(100.0);
numbers.add(800.9);

//String等非Number類型就不行
//printElementUpbound(names);

//此時Double類型也不行
//printElementDownbound(numbers);

//泛型值只能是Integer及其父類類型,所以Double/Float/String等類型都不可以
printElementDownbound(ages);

}

//下限通配符的使用,這里的泛型值只能是Integer及其父類類型
public static void printElementDownbound(List<? super Integer> data) {
for(int i=0;i<data.size();i++) {
System.out.println(data.getClass().getSimpleName()+"--data: " + data.get(i));
}
}
}

  在這個例子中,printElementDownbound方法中的集合泛型,可以是Integer或其父類型,即類型下限是Integer,除此之外的其他類型都不行。也就是說,我們只能使用Integer或其父類作為類型參數,泛型類型的下限是Integer,這就是下限通配符的含義。

  7. 的區別

  在這里,小編要給大家再總結一下的區別:

  ● 允許調用T get()這樣的讀方法來獲取T對象的引用,但不允許調用set(T)這樣的寫方法來傳入T的引用(傳入null除外);

  ● 允許調用set(T)這樣的寫方法傳入T對象的引用,但不允許調用T get()這樣的讀方法來獲取T對象的引用(獲取Object除外)。

  ● 即允許讀不允許寫,允許寫不允許讀。

  大家注意,無論是未限定通配符、上限通配符還是下限通配符,我們既可以在方法中使用,也可以在類或接口中使用。

  三. 泛型擦除

  我們在學習泛型時,除了要掌握上面這些泛型類、泛型接口、泛型方法以及通配符等內容之外,還要學習泛型擦除的相關內容。那么什么是泛型擦除呢?我們繼續往下學習吧。

  1.簡介

  所謂的泛型擦除(Type Erasure),就是指在編譯時,JVM編譯器會將所有的泛型信息都擦除掉,變成原始類型,一般是將泛型的類型參數替換成具體類型的上限或下限(如果沒有指定上界,則默認為Object)。

  換句話說,雖然我們在代碼中使用了泛型,但在編譯后,所有的泛型類型都會被擦除掉,轉而使用其對應的原始類型。這就是Java泛型的底層實現原理。這樣設計的目的是為了兼容舊的JDK版本,使得Java具有了較好的向后兼容性,舊的非泛型代碼可以直接使用泛型類庫,而不需要進行任何修改。同時,Java也提供了反射機制來操作泛型類型,使得泛型類型在某些情況下還是可以被獲取到的,所以即使有泛型擦除,仍然也不會太影響Java虛擬機的運行時效率。

  比如,在我們定義一個泛型類時,我們會使用泛型類型參數來代替具體的類型,好比下面這個例子:

public class Box<T> {
private T content;

public Box(T content) {
this.content = content;
}

public T getContent() {
return content;
}

public void setContent(T content) {
this.content = content;
}
}

  2.泛型擦除帶來的限制

  在編譯之后,這個泛型類的類型參數T就會被擦除,成為其對應的原始類型Object。這也意味著,我們無法在運行時獲取到泛型的實際類型參數,所以泛型擦除的使用會有一些限制。首先由于泛型類型參數被擦除了,因此我們在運行時就無法獲得泛型類型參數的信息。例如,如果我們有一個List類型的變量,在運行時我們就無法獲得這個List集合中的元素類型是Integer。另一個限制是在使用泛型類型時,還需要注意類型安全性。在編譯階段,編譯器會檢查泛型類型的類型安全性;但在運行階段,由于泛型類型參數被擦除了,因此就無法保證類型安全性了。泛型擦除的限制,主要表現在以下幾個方面:

  無法使用基本類型實例化類型參數;

  無法在運行時獲取泛型類型信息;

  泛型類型參數不能用于靜態變量或靜態方法;

  不能實例化T類型。

  接下來小編再給大家具體分析一下這些限制。

  2.1 無法使用基本類型實例化類型參數

  Java泛型中的類型參數不能是基本類型,只能是類或接口類型。例如,以下代碼在編譯階段會出錯,無法通過編譯:  

List<int> list = new ArrayList<int>();

  正確的寫法是使用基本類型對應的包裝類型,如下所示:  

List<Integer> list = new ArrayList<Integer>();

  2.2 無法在運行時獲取泛型類型信息

  由于泛型擦除的存在,導致我們在程序運行時無法獲取泛型類型的信息。例如,以下代碼在運行時就無法獲取List的元素類型:  

List<String> list = new ArrayList<String>(); 
Class<?> clazz = list.getClass();
Type type = clazz.getGenericSuperclass();
// 輸出:class java.util.ArrayList<E>
System.out.println(type);

  在輸出的結果中,我們只能得到ArrayList的類型信息,而無法獲取到集合中具體的泛型類型信息,也就是獲取不到String的信息。但如果我們就是想在運行時獲取到泛型的實際類型參數,其實可以參考以下方式進行實現: 

public class Box<T> {
private T content;

public Box(T content) {
this.content = content;
}

public T getContent() {
return content;
}

public void setContent(T content) {
this.content = content;
}

public Class<?> getContentType() {
return content.getClass();
}
}

  在上面的代碼中,我們新增了一個方法 getContentType(),該方法用于返回泛型類型參數的實際字節碼類型,以后我們通過調用這個方法,就可以間接地獲取到泛型類型的信息了。

  2.3 泛型類型參數不能用于靜態變量或靜態方法

  由于泛型類型參數是在實例化對象時才被確定的,因此不能在靜態變量或靜態方法中使用泛型類型參數。例如,以下代碼是無法編譯通過的:  

public class MyClass<T> {   
//這樣的代碼編譯不通過
private static T value;

public static void setValue(T value) {
MyClass.value = value;
}
}

  正確的寫法是使用一個實際類型來代替泛型類型參數:  

public class MyClass {     
private static String value;
public static void setValue(String value) {
MyClass.value = value;
}
}

  2.4 不能實例化T類型

  比如在下面的案例中:  

public class MyClass<T> {
private T first;
private T last;

public MyClass() {
first = new T();
last = new T();
}
}

  上述代碼無法通過編譯,因為構造方法的兩行語句: 

first = new T(); 
last = new T();

  擦拭后實際上變成了:  

first = new Object(); 
last = new Object();

  這樣一來,創建new MyClass()和創建new MyClass()就變成了Object,編譯器會阻止這種類型不對的代碼。如果我們想對泛型T進行實例化,需要借助Class參數并集合反射技術來實現,且在使用時也必須傳入Class。

  四. 結語

  不過,盡管泛型擦除有一些限制,但泛型仍然不失為一種強大的編程工具,它可以提高代碼的可讀性和可維護性。通過合理地使用泛型,我們可以在編譯時進行類型檢查,避免類型轉換的錯誤和運行時異常,從而提高了代碼的安全性和可靠性。同時,我們也需要了解Java泛型擦除的限制,以便在實際應用中做出正確的決策。

聲明:本站稿件版權均屬千鋒教育所有,未經許可不得擅自轉載。
10年以上業內強師集結,手把手帶你蛻變精英
請您保持通訊暢通,專屬學習老師24小時內將與您1V1溝通
免費領取
今日已有369人領取成功
劉同學 138****2860 剛剛成功領取
王同學 131****2015 剛剛成功領取
張同學 133****4652 剛剛成功領取
李同學 135****8607 剛剛成功領取
楊同學 132****5667 剛剛成功領取
岳同學 134****6652 剛剛成功領取
梁同學 157****2950 剛剛成功領取
劉同學 189****1015 剛剛成功領取
張同學 155****4678 剛剛成功領取
鄒同學 139****2907 剛剛成功領取
董同學 138****2867 剛剛成功領取
周同學 136****3602 剛剛成功領取
相關推薦HOT
色综合老司机第九色激情 _中文字幕日韩av资源站_国产+人+亚洲_久久久精品影院_久久久视频免费观看_欧美激情亚洲自拍_亚洲成av人片在线观看香蕉_热草久综合在线_欧美极品第一页_2020国产精品自拍
一区二区在线观看不卡| 欧美日韩在线播| 成人动漫精品一区二区| 伦理电影国产精品| 欧美精品一区二区三区蜜桃视频| 欧美午夜一区二区三区| 成人免费毛片片v| 日韩免费在线观看| 在线亚洲高清视频| 一本大道久久a久久精二百| 国产黄色91视频| 精品写真视频在线观看| 亚洲成人激情av| 中文字幕在线观看一区| caoporn国产一区二区| 亚洲电影中文字幕在线观看| 欧美日韩综合一区| 国产高清精品网站| 亚洲午夜三级在线| 91国模大尺度私拍在线视频| 欧美激情在线一区二区三区| 在线视频欧美精品| 日韩午夜中文字幕| av成人老司机| 91在线视频18| 日韩欧美区一区二| 亚洲国产精品av| 午夜伦欧美伦电影理论片| 日韩 欧美一区二区三区| 日韩电影在线一区二区三区| 亚洲va欧美va天堂v国产综合| 国产日韩欧美激情| 91麻豆精品国产91久久久久| 日本二三区不卡| 日本成人在线网站| 亚洲一线二线三线视频| 日韩一区二区电影网| 午夜视频一区二区三区| 成人美女在线观看| 婷婷中文字幕一区三区| 一区二区三区产品免费精品久久75| 日本乱人伦aⅴ精品| 欧美性色综合网| 欧美电影影音先锋| 国产精品视频在线看| www.亚洲色图| 欧美一a一片一级一片| 久久免费视频色| 五月天视频一区| 色婷婷久久久亚洲一区二区三区| 91成人在线免费观看| 欧美性xxxxx极品少妇| 色婷婷激情一区二区三区| 欧美色图第一页| 制服丝袜中文字幕一区| 亚洲精品一线二线三线无人区| 激情av综合网| 日日摸夜夜添夜夜添国产精品| 国精产品一区一区三区mba视频 | 九色综合狠狠综合久久| 日韩三级高清在线| 亚洲宅男天堂在线观看无病毒| 奇米综合一区二区三区精品视频| 欧美军同video69gay| 中文字幕一区二区三中文字幕| 伊人一区二区三区| 国产亚洲综合在线| 国产精品乱码人人做人人爱| 国产一区二区久久| 国产日韩欧美一区二区三区乱码| 国产在线不卡一卡二卡三卡四卡| 国产精品色在线观看| 9色porny自拍视频一区二区| 中文字幕在线一区二区三区| 99国产欧美另类久久久精品| 日本中文一区二区三区| 国产在线播放一区| 色综合久久久久综合体桃花网| 26uuu色噜噜精品一区| 亚洲综合免费观看高清完整版 | 国产精品女上位| 久久成人久久爱| 精品成人a区在线观看| 日本高清无吗v一区| 91国产丝袜在线播放| 日韩av在线播放中文字幕| 日韩一二三四区| 经典三级在线一区| 国产精品免费免费| 欧美一级黄色录像| 国产自产高清不卡| 欧美国产精品一区二区| 国产精品一区二区三区网站| 国产精品不卡一区二区三区| 4438x亚洲最大成人网| 国内精品不卡在线| 国产精品久久福利| 91精品国产欧美一区二区成人| 国产高清亚洲一区| 日韩精品久久久久久| 国产欧美日韩在线看| 欧美视频在线播放| 欧美日韩免费观看一区二区三区| 国产不卡视频在线播放| 狠狠色狠狠色综合日日91app| 国产欧美日韩另类一区| 91极品美女在线| 国产精品一区二区你懂的| 视频一区在线播放| 亚洲精品高清在线观看| 久久99久久99| 日韩精品欧美精品| 亚洲一区二区三区在线看| 一区二区三区日韩欧美精品| 亚洲h在线观看| 国产一区不卡视频| 蜜桃av一区二区三区| 亚洲午夜在线视频| 亚洲综合激情网| 国产精品久久夜| 一区在线播放视频| 国产精品毛片久久久久久久| 2021国产精品久久精品| 久久久亚洲精品石原莉奈| 日本一区二区三区久久久久久久久不| 欧美一区二区成人| 777午夜精品免费视频| youjizz久久| 欧美系列亚洲系列| 久久综合九色欧美综合狠狠| 日本一区二区在线不卡| 国产日韩一级二级三级| 久久久久久久精| 亚洲视频在线一区观看| 亚洲第一二三四区| 国产在线麻豆精品观看| 91浏览器打开| 91猫先生在线| 成人av网站在线观看免费| 欧美日韩中文精品| 久久精品视频免费| 毛片基地黄久久久久久天堂| 精久久久久久久久久久| 91成人在线精品| 国产日韩精品一区二区三区在线| 亚洲尤物在线视频观看| 亚洲电影欧美电影有声小说| 婷婷一区二区三区| 91丨九色丨蝌蚪丨老版| 日韩精品一区二区三区四区 | 日韩精品一区二区三区三区免费| 日本一区二区三区在线不卡| 一区二区三区欧美激情| 久久成人久久爱| 欧美日韩精品专区| 国产欧美久久久精品影院| 韩日精品视频一区| 日韩一区二区免费电影| 一区二区欧美在线观看| 国产精品一线二线三线| 欧美日韩一级大片网址| 亚洲视频在线一区二区| 9人人澡人人爽人人精品| 久久久久久亚洲综合影院红桃| 日本视频在线一区| 精品国产一区二区三区久久影院| 午夜国产精品一区| 欧美精品在线视频| 夜夜精品浪潮av一区二区三区| 成人午夜在线视频| 中文幕一区二区三区久久蜜桃| 爽爽淫人综合网网站| 在线视频一区二区三| 午夜精品久久久久久久久| 日韩一区二区三区免费观看| 精品一区二区日韩| 国产精品美女久久久久久久久久久 | 色综合久久久网| 亚洲国产精品久久艾草纯爱| 欧美日韩免费高清一区色橹橹 | 亚洲裸体xxx| 欧美体内she精高潮| 国产一区二区三区四区五区美女 | 91麻豆精品国产91久久久| 秋霞午夜鲁丝一区二区老狼| 久久久青草青青国产亚洲免观| 国产一区不卡精品| **欧美大码日韩| 欧美成人一区二区三区片免费 | 国产69精品久久777的优势| 亚洲欧美日韩一区| 久久网站最新地址| 7777精品伊人久久久大香线蕉经典版下载| 国产一区二区视频在线播放| 天天做天天摸天天爽国产一区| 欧美日韩不卡视频| 欧美伊人久久久久久午夜久久久久| 国产中文字幕一区| 久久精品人人做人人爽人人| 色婷婷激情综合|