Java泛型初學(xué)者之上、下界通配符的深入理解
泛型的由來(lái)
為什么需要泛型
Java的數(shù)據(jù)類型一般都是在定義時(shí)就需要確定,這種強(qiáng)制的好處就是類型安全,不會(huì)出現(xiàn)像弄一個(gè)ClassCastException的數(shù)據(jù)給jvm,數(shù)據(jù)安全那么執(zhí)行的class就會(huì)很穩(wěn)定。但是假如說(shuō)我不知道這個(gè)參數(shù)要傳什么類型的,因?yàn)楣拘枨笤谧?,如果?xiě)死的那就只能便以此需求就改一次,很麻煩。sun公司也注意到這個(gè)問(wèn)題,這樣會(huì)讓代碼的靈活性降低,他們就研究出了泛型。
泛型初識(shí)
什么是泛型,可以字面理解就是一個(gè)泛泛的類型,他是不確定的,在Java代碼編譯的時(shí)候用泛型是不會(huì)出錯(cuò)的,而在運(yùn)行期時(shí)就會(huì)報(bào)錯(cuò),說(shuō)你這種第一是不合理的。這是為什么呢。因?yàn)闉榱颂岣哽`活性,就在編譯時(shí)期將條件放寬,但是泛型一定要在運(yùn)行的時(shí)候告訴jvm你給我的數(shù)據(jù)到底是什么類型的,否則jvm會(huì)是懵逼的。所以泛型的好處就是將類型的靈活性提高,也只是在Java語(yǔ)法的基礎(chǔ)上提高,不過(guò)泛型還是比較實(shí)用的。
何時(shí)使用泛型
泛型的應(yīng)用場(chǎng)景就是應(yīng)用在模型(可以理解為存儲(chǔ)數(shù)據(jù)的盒子),我為了這個(gè)盒子適用更多的地方我就用將需要存入的數(shù)據(jù)用一個(gè)泛型表示,當(dāng)然可以傳入多值。如果是相同類型的對(duì)象就用一個(gè)泛型的數(shù)組比較好,學(xué)過(guò)集合的小伙伴應(yīng)該都知道,沒(méi)學(xué)過(guò)的那你應(yīng)該補(bǔ)補(bǔ)課了。
泛型的語(yǔ)法
public class A<T extends B>{
T t;
}
泛型的缺點(diǎn)或者為什么需要上、下邊界
泛型的雖然強(qiáng)大,但是世界上任何東西東部是完美的。它也有缺陷。比如說(shuō)我有一個(gè)盒子我想裝蘋(píng)果,但是我還可能想裝香蕉那怎么辦。那還不好說(shuō),在給一個(gè)參數(shù)不就行了,那十個(gè)呢,二十個(gè)呢。em....的確是。如果說(shuō)我們想裝的東西都屬于一個(gè)類并且只要是這個(gè)類的子類就可以裝。這個(gè)想法sun為我們想好了。那就是用上邊界通配符。語(yǔ)法是 T是泛型,M是T的父類。我們就定義一個(gè)水果類(Fruit),盛裝就容器就是盤(pán)子(Dish),現(xiàn)在我們就可以裝任何水果了,不錯(cuò)吧!
上邊界Java代碼
public class Dish<T extends Fruit>{
private T fruitChild;
public Dish(T fruitChild){
this.fruitChild = fruitChild;
}
public T getFruitChild(){
return fruitChild;
}
public void setFruitChild(T f){
this.fruitChild = f;
}
public static void main(String[] args){
Dish dish = new Dish<apple>();
Apple apple = new apple(); //apple must be Fruit child;
dish.setFruitChild(apple);
system.out.printf(dish.getFruitChild);
}
}
下邊界Java代碼
public class Dish<T super Apple>{
private T appleFather;
public Dish(T appleFather){
this.appleFather = appleFather;
}
public T getAppleFather(){
return appleFather;
}
public void setAppleFather(T f){
this.appleFather = f;
}
public static void main(String[] args){
Dish dish = new Dish<Fruit>();
Fruit fruit = new Fruit(); //fruit must be apple son;
dish.setAppleFather(fruit);
system.out.printf(dish.getAppleFather);
}
}
什么是上邊界通配符
當(dāng)泛型T給定形如 的A類型到A類型任何子類的限制域,可以匹配任何在此限制域中的類型,此種表示叫上邊界通配符。
上邊界通配符理解

什么是下邊界通配符
當(dāng)泛型T給定形如 的A類型到A類型任何父類的限制域,可以匹配任何在此限制域中的類型,此種表示叫下邊界通配符。
下邊界通配符理解#

上下邊界通配符的缺點(diǎn)
上界<? extends T>不能往里存,只能往外取。
解釋
因?yàn)榫幾g器只知道傳入的是T的子類,但具體是哪一個(gè)編譯器不知道,他只標(biāo)注了一個(gè)占位符,當(dāng)?傳過(guò)來(lái)時(shí),他不知道這能不能和占位符匹配,所以set不了。
下界<? super T>不影響往里存,但往外取只能放在Object對(duì)象里。
解釋
因?yàn)橄逻吔缫呀?jīng)限制了?的粒度,他只可能是T本身或者是T的父類。我們想想,我想要一個(gè)T,你卻返回給我一個(gè)比T小的Object,這樣我們就因?yàn)榫葥p失而拿不到想要的數(shù)據(jù)了。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
手把手教你使用IDEA創(chuàng)建多模塊(maven)項(xiàng)目
這篇文章主要給大家介紹了關(guān)于如何使用IDEA創(chuàng)建多模塊(maven)項(xiàng)目的相關(guān)資料,文中通過(guò)圖文以及實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-07-07
Java Semaphore實(shí)現(xiàn)高并發(fā)場(chǎng)景下的流量控制
在java開(kāi)發(fā)的工作中是否會(huì)出現(xiàn)這樣的場(chǎng)景,你需要實(shí)現(xiàn)一些異步運(yùn)行的任務(wù),該任務(wù)可能存在消耗大量?jī)?nèi)存的情況,所以需要對(duì)任務(wù)進(jìn)行并發(fā)控制。本文將介紹通過(guò)Semaphore類優(yōu)雅的實(shí)現(xiàn)并發(fā)控制,感興趣的可以了解一下2021-12-12
如何基于spring security實(shí)現(xiàn)在線用戶統(tǒng)計(jì)
這篇文章主要介紹了如何基于spring security實(shí)現(xiàn)在線用戶統(tǒng)計(jì),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06
SpringBoot+Response如何統(tǒng)一返回result結(jié)果集
這篇文章主要介紹了SpringBoot+Response如何統(tǒng)一返回result結(jié)果集,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05
RabbitMQ的安裝集群、鏡像隊(duì)列配置規(guī)劃
RabbitMQ 集群是一個(gè)或多個(gè)節(jié)點(diǎn)的邏輯分組,每個(gè)節(jié)點(diǎn)共享用戶、虛擬主機(jī)、隊(duì)列、流、交換機(jī)、綁定、運(yùn)行時(shí)參數(shù)和其他分布式狀態(tài),本文給大家介紹RabbitMQ的安裝集群、鏡像隊(duì)列配置規(guī)劃,感興趣的朋友一起看看吧2025-09-09

