淺談Java泛型讓聲明方法返回子類型的方法
泛型典型的使用場景是集合。考慮到大多數(shù)情況下集合是同質(zhì)的(同一類型),通過聲明參數(shù)類型,可免去類型轉(zhuǎn)換的麻煩。本文將討論本人閱讀Spring Security源碼時遇到的一個關(guān)于泛型遞歸模式的問題。
聲明方法返回子類型
在Spring Security的源碼里有一個ProviderManagerBuilder接口,聲明如下
public interface ProviderManagerBuilder<B extends ProviderManagerBuilder<B>> extends SecurityBuilder<AuthenticationManager> { B authenticationProvider(AuthenticationProvider authenticationProvider); }
其實現(xiàn)類AuthenticationManagerBuilder
public class AuthenticationManagerBuilder extends AbstractConfiguredSecurityBuilder<AuthenticationManager, AuthenticationManagerBuilder> implements ProviderManagerBuilder<AuthenticationManagerBuilder> { //... public AuthenticationManagerBuilder authenticationProvider( AuthenticationProvider authenticationProvider) { this.authenticationProviders.add(authenticationProvider); return this; } //... }
上面有很多干擾項,我們來簡化一下
接口A定義如下
public interface A<T extends A<T>> { T add(); }
說明:A接口只有一個add方法,返回泛型T。T的聲明有些饒<T extends A<T>>。
A接口的實現(xiàn)類B
public class B implements A<B> { @Override public B add() { return null; } }
注意,此處類B里的add方法返回類型B。也就是說,接口A里聲明的方法時并不知道子類型B的存在,通過繼承和泛型,可以放返回值動態(tài)的適配子類型,這一切都要歸功于<T extends A<T>>
泛型遞歸模式(Recurring Generic Pattern)
public interface A public abstract class Enum<E extends Enum<E>> implements Comparable<E>, Serializable { //... }
java所有的枚舉類型都隱式的繼承java.lang.Enum,不允許通過現(xiàn)實的繼承聲明枚舉類型,甚至集成java.lang.Enum也是編譯器所不允許的。
假設(shè)有一個枚舉類StatusCode,其等價的聲明如下
public class StatusCode extends Enum<StatusCode>
現(xiàn)在我們來驗證一下泛型約束,
1. 因為Enum<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不行么?
因為Enum<E>實現(xiàn)了Comparable< E>接口,該接口有一個compareTo方法
public int compareTo(E o) {}
<E extend Enum> 強制約束了進行`compareTo`的調(diào)用對象類型和參數(shù)類型都嚴格一致,不會出現(xiàn)子類和超類或者兄弟類之間的比較。
泛型遞歸模式與繼承
泛型遞歸模式interface A<T extend A<T>>用于約束參數(shù)類型T,要求其為類型A的子類。
考慮到繼承和實現(xiàn)B implements A< B>,參數(shù)類型和實體類型是一致的。這樣類A中方法簽名里涉及到參數(shù)類型T的地方,在實現(xiàn)類里會為實現(xiàn)類本身,這讓類型系統(tǒng)更加的嚴謹。
以上這篇淺談Java泛型讓聲明方法返回子類型的方法就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
解決RestTemplate 請求接收自定義400+ 或500+錯誤
這篇文章主要介紹了解決RestTemplate 請求接收自定義400+ 或500+錯誤,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08Nacos docker單機模式部署實現(xiàn)過程詳解
這篇文章主要介紹了Nacos docker單機模式部署實現(xiàn)過程詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-09-09淺談一下RabbitMQ、Kafka和RocketMQ消息中間件對比
這篇文章主要介紹了淺談一下RabbitMQ、Kafka和RocketMQ消息中間件對比,消息中間件屬于分布式系統(tǒng)中一個字系統(tǒng),關(guān)注于數(shù)據(jù)的發(fā)送和接收,利用高效可靠的異步信息傳遞機制對分布式系統(tǒng)中的其余各個子系統(tǒng)進行集成,需要的朋友可以參考下2023-05-05java多線程join()方法的作用和實現(xiàn)原理解析(應(yīng)用場景)
join方法主要是用于將當(dāng)前線程掛起,等待其他線程結(jié)束后在執(zhí)行當(dāng)前線程,本文通過應(yīng)用場景分析代碼示例講解java多線程join()方法的作用和實現(xiàn)原理,感興趣的朋友一起看看吧2021-07-07Java使用字節(jié)流實現(xiàn)圖片音頻的復(fù)制
今天帶大家學(xué)習(xí)Java的相關(guān)知識,文章圍繞著Java如何使用字節(jié)流實現(xiàn)圖片音頻的復(fù)制展開,文中有非常詳細的介紹,需要的朋友可以參考下2021-06-06