关于通配泛型类型有几种_5.7 泛型通配符和类型参数的范围
本節(jié)先講解如何限制類型參數(shù)的范圍,再講解通配符(?)。
類型參數(shù)的范圍
在泛型中,如果不對類型參數(shù)加以限制,它就可以接受任意的數(shù)據(jù)類型,只要它是被定義過的。但是,很多時候我們只需要一部分?jǐn)?shù)據(jù)類型就夠了,用戶傳遞其他數(shù)據(jù)類型可能會引起錯誤。例如,編寫一個泛型函數(shù)用于返回不同類型數(shù)組(Integer 數(shù)組、Double 數(shù)組等)中的最大值:public??T?getMax(T?array[]){
T?max?=?null;
for(T?element?:?array){
max?=?element.doubleValue()?>?max.doubleValue()???element?:?max;
}
return?max;
}
上面的代碼會報(bào)錯,doubleValue() 是 Number 類及其子類的方法,不是所有的類都有該方法,所以我們要限制類型參數(shù) T,讓它只能接受 Number 及其子類(Integer、Double、Character 等)。
通過 extends 關(guān)鍵字可以限制泛型的類型的上限,改進(jìn)上面的代碼:public??T?getMax(T?array[]){
T?max?=?null;
for(T?element?:?array){
max?=?element.doubleValue()?>?max.doubleValue()???element?:?max;
}
return?max;
}
表示 T 只接受 Number 及其子類,傳入其他類型的數(shù)據(jù)會報(bào)錯。這里的限定使用關(guān)鍵字?extends,后面可以是類也可以是接口。如果是類,只能有一個;但是接口可以有多個,并以“&”分隔,例如 。
這里的 extends 關(guān)鍵字已不再是繼承的含義了,應(yīng)該理解為 T 是繼承自 Number 類的類型,或者 T 是實(shí)現(xiàn)了 XX 接口的類型。
通配符(?)
上一節(jié)的例子中提到要定義一個泛型類來表示坐標(biāo),坐標(biāo)可以是整數(shù)、小數(shù)或字符串,請看下面的代碼:class?Point{
T1?x;
T2?y;
public?T1?getX()?{
return?x;
}
public?void?setX(T1?x)?{
this.x?=?x;
}
public?T2?getY()?{
return?y;
}
public?void?setY(T2?y)?{
this.y?=?y;
}
}
現(xiàn)在要求在類的外部定義一個 printPoint()?方法用于輸出坐標(biāo),怎么辦呢?
可以這樣來定義方法:public?void?printPoint(Point?p){
System.out.println("This?point?is:?"?+?p.getX()?+?",?"?+?p.getY());
}
我們知道,如果在使用泛型時沒有指名具體的數(shù)據(jù)類型,就會擦除泛型類型,并向上轉(zhuǎn)型為 Object,這與不使用泛型沒什么兩樣。上面的代碼沒有指明數(shù)據(jù)類型,相當(dāng)于:
public?void?printPoint(Point?p){
System.out.println("This?point?is:?"?+?p.getX()?+?",?"?+?p.getY());
}
為了避免類型擦除,可以使用通配符(?):
public?void?printPoint(Point,??>?p){
System.out.println("This?point?is:?"?+?p.getX()?+?",?"?+?p.getY());
}
通配符(?)可以表示任意的數(shù)據(jù)類型。將代碼補(bǔ)充完整:
public?class?Demo?{
public?static?void?main(String[]?args){
Point?p1?=?new?Point();
p1.setX(10);
p1.setY(20);
printPoint(p1);
Point?p2?=?new?Point();
p2.setX("東京180度");
p2.setY("北緯210度");
printPoint(p2);
}
public?static?void?printPoint(Point,??>?p){??//?使用通配符
System.out.println("This?point?is:?"?+?p.getX()?+?",?"?+?p.getY());
}
}
class?Point{
T1?x;
T2?y;
public?T1?getX()?{
return?x;
}
public?void?setX(T1?x)?{
this.x?=?x;
}
public?T2?getY()?{
return?y;
}
public?void?setY(T2?y)?{
this.y?=?y;
}
}
運(yùn)行結(jié)果:
This point is: 10, 20
This point is: 東京180度, 北緯210度
但是,數(shù)字坐標(biāo)與字符串坐標(biāo)又有區(qū)別:數(shù)字可以表示x軸或y軸的坐標(biāo),字符串可以表示地球經(jīng)緯度。現(xiàn)在又要求定義兩個方法分別處理不同的坐標(biāo),一個方法只能接受數(shù)字類型的坐標(biāo),另一個方法只能接受字符串類型的坐標(biāo),怎么辦呢?
這個問題的關(guān)鍵是要限制類型參數(shù)的范圍,請先看下面的代碼:
public?class?Demo?{
public?static?void?main(String[]?args){
Point?p1?=?new?Point();
p1.setX(10);
p1.setY(20);
printNumPoint(p1);
Point?p2?=?new?Point();
p2.setX("東京180度");
p2.setY("北緯210度");
printStrPoint(p2);
}
//?借助通配符限制泛型的范圍
public?static?void?printNumPoint(Point?extends?Number,???extends?Number>?p){
System.out.println("x:?"?+?p.getX()?+?",?y:?"?+?p.getY());
}
public?static?void?printStrPoint(Point?extends?String,???extends?String>?p){
System.out.println("GPS:?"?+?p.getX()?+?","?+?p.getY());
}
}
class?Point{
T1?x;
T2?y;
public?T1?getX()?{
return?x;
}
public?void?setX(T1?x)?{
this.x?=?x;
}
public?T2?getY()?{
return?y;
}
public?void?setY(T2?y)?{
this.y?=?y;
}
}
運(yùn)行結(jié)果:
x: 10, y: 20
GPS: 東京180度,北緯210度
? extends Number 表示泛型的類型參數(shù)只能是 Number 及其子類,? extends String 也一樣,這與定義泛型類或泛型方法時限制類型參數(shù)的范圍類似。
不過,使用通配符(?)不但可以限制類型的上限,還可以限制下限。限制下限使用 super 關(guān)鍵字,例如 super Number> 表示只能接受 Number 及其父類。
注意:一般的項(xiàng)目中很少會去設(shè)計(jì)泛型,這里主要是讓讀者學(xué)會如何使用,為后面的教程做鋪墊。
總結(jié)
以上是生活随笔為你收集整理的关于通配泛型类型有几种_5.7 泛型通配符和类型参数的范围的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 模拟撞击_模玩资讯:EPOCH 汽车撞击
- 下一篇: getlab如何编辑提交时显示的用户名_