泛型典型的使用場景是集合。考慮到大多數(shù)情況下集合是同質(zhì)的(同一類型),通過聲明參數(shù)類型,可免去類型轉(zhuǎn)換的麻煩。本文將討論本人閱讀Spring Security源碼時(shí)遇到的一個(gè)關(guān)于泛型遞歸模式的問題。
聲明方法返回子類型
在Spring Security的源碼里有一個(gè)ProviderManagerBuilder接口,聲明如下
1
2
3
|
public interface ProviderManagerBuilder<B extends ProviderManagerBuilder<B>> extends SecurityBuilder<AuthenticationManager> { B authenticationProvider(AuthenticationProvider authenticationProvider); } |
其實(shí)現(xiàn)類AuthenticationManagerBuilder
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public class AuthenticationManagerBuilder extends AbstractConfiguredSecurityBuilder<AuthenticationManager, AuthenticationManagerBuilder> implements ProviderManagerBuilder<AuthenticationManagerBuilder> { //... public AuthenticationManagerBuilder authenticationProvider( AuthenticationProvider authenticationProvider) { this .authenticationProviders.add(authenticationProvider); return this ; } //... } |
上面有很多干擾項(xiàng),我們來簡化一下
接口A定義如下
1
2
3
4
5
|
public interface A<T extends A<T>> { T add(); } |
說明:A接口只有一個(gè)add方法,返回泛型T。T的聲明有些饒<T extends A<T>>。
A接口的實(shí)現(xiàn)類B
1
2
3
4
5
6
7
8
|
public class B implements A<B> { @Override public B add() { return null ; } } |
注意,此處類B里的add方法返回類型B。也就是說,接口A里聲明的方法時(shí)并不知道子類型B的存在,通過繼承和泛型,可以放返回值動(dòng)態(tài)的適配子類型,這一切都要?dú)w功于<T extends A<T>>
泛型遞歸模式(Recurring Generic Pattern)
1
2
3
4
5
6
|
public interface A public abstract class Enum<E extends Enum<E>> implements Comparable<E>, Serializable { //... } |
java所有的枚舉類型都隱式的繼承java.lang.Enum,不允許通過現(xiàn)實(shí)的繼承聲明枚舉類型,甚至集成java.lang.Enum也是編譯器所不允許的。
假設(shè)有一個(gè)枚舉類StatusCode,其等價(jià)的聲明如下
public class StatusCode extends Enum<StatusCode>
現(xiàn)在我們來驗(yàn)證一下泛型約束,
1. 因?yàn)镋num<StatusCode>,所以E=StatusCode;
2. 根據(jù)<E extend Enum<E>> 和 E=StatusCode 可得,<StatusCode extend Enum<StatusCode>>;
3. 由于public class StatusCode extends Enum<StatusCode>第二步的結(jié)論顯然成立。
為什么Enum的聲明這么繞?直接Enum不行么?
因?yàn)镋num<E>實(shí)現(xiàn)了Comparable< E>接口,該接口有一個(gè)compareTo方法
public int compareTo(E o) {}
<E extend Enum> 強(qiáng)制約束了進(jìn)行`compareTo`的調(diào)用對(duì)象類型和參數(shù)類型都嚴(yán)格一致,不會(huì)出現(xiàn)子類和超類或者兄弟類之間的比較。
泛型遞歸模式與繼承
泛型遞歸模式interface A<T extend A<T>>用于約束參數(shù)類型T,要求其為類型A的子類。
考慮到繼承和實(shí)現(xiàn)B implements A< B>,參數(shù)類型和實(shí)體類型是一致的。這樣類A中方法簽名里涉及到參數(shù)類型T的地方,在實(shí)現(xiàn)類里會(huì)為實(shí)現(xiàn)類本身,這讓類型系統(tǒng)更加的嚴(yán)謹(jǐn)。
以上這篇淺談Java泛型讓聲明方法返回子類型的方法就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持服務(wù)器之家。