Spring中初始化泛型類的方法實例
首先來看下在 Java 中對于泛型類型,比如這樣簡單的類定義
class Processor<T> {}
如果直接初始化時要指定具體類型的話,我們可以這么寫
Processor<String> processor = new Processor<>(); //Java 7 及以上版本
Spring 對基本泛型的初始化
如果我們要用 Spring 容器來初始化這個類,比如給上面那個類加個 @Named 注解
@Named
class Processor<T> {
}
這時候我們通過 beanFactory.getBean(Processor.class) 得到的是一個什么樣的實例呢?Spring 怎么知道要指定什么具體類型呢?很簡單,任何不確定的情況都是 Object。所以通過容器得到的 Processor 實例相當(dāng)于用下面代碼構(gòu)造出來的
Processor processor = new Processor(); //更準(zhǔn)確來講是 Processor<Object> processor = new Processor<>();
再進(jìn)一步,對于有上限約束的泛型定義,Spring 才如何應(yīng)對呢?像
@Named
class Processor<T extends Number> {
}
類似的,class Processor<T> 相當(dāng)于 class Processor<T extends Object> , 因此 Spring 在具體類型未明的情況下也是要用最頂層可接受類型,Spring 將會針對上面的代碼實例出下面的對象
Processor<Number> processor = new Processor<>();
再復(fù)雜一些,泛型的子類型仍然是泛型的情況,如下代碼
首先定義了一個泛型接口
public interface Service<T> {
String process(T t);
}
然后要求 Spring 容器來初始下面的 NumberService 實例
@Named
public class NumberService<R extends Number> implements Service<R> {
@Override
public String process(R number) {
return "Process Number: " + number;
}
}
Spring 在初始化 NumberService 實例同樣是要取用最頂層可接受類型,通過下面的代碼來初始化
NumberService<Number> numberService = new NumberService<>();
再終極一些,泛型類型并且類型也是泛型的,Spring 該如何拿捏?
@Named
public class Processor<T> {
@Inject
Private Service<T> service;
}
此時 Spring 該如何確定上面的類型 T 呢?因為有了 Service<T> service 屬性的存在而不能再籠統(tǒng)的想像 Spring 會采用下面的代碼來初始化 Processor 實例
Processor<Object> processor = new Processor<>();
而是 Processor 的具體類型必須通過被注入的 Service<T> 實例的具體類型來推斷的,這就取決于在 Spring 容器中存在什么樣的 Service<T> 實例。舉兩個例子
如果 Spring 中有初始化
@Named
public class StringService implements Service<String> {
@Override
public String process(String string) {
return "Process String: " + string;
}
}
那么前面的 Processor<T> 實例就相當(dāng)于
Processor<String> processor = new Processor<>(); processor.service = new StringService();
如果 Spring 中初始化的 Service<T> 是前面那個 NumberService<R extends Number> implements Service<R> , 那么 Spring 容器中的 Processor<T> 實例相當(dāng)于
Processor<Number> processor = new Processor<>(); processor.service = new NumberService<Number>();
那如果前面的 NumberService 和 StringService 同時在 Spring 容器中注冊了呢?Spring 同樣要為難了,在沒有 @Primary 的情況下無法確定使用哪個實例來注入 Service<T> service 屬性了,出現(xiàn)類似錯誤
2016-12-09 00:56:50.922 WARN 4950 --- [ main] s.c.a.AnnotationConfigApplicationContext : Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'processor': Unsatisfied dependency expressed through field 'service'; nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'cc.unmi.Service<?>' available: expected single matching bean but found 2: numberService,stringService 2016-12-09 00:56:50.941 ERROR 4950 --- [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPLICATION FAILED TO START *************************** Description: Field service in cc.unmi.Processor required a single bean, but 2 were found: - numberService: defined in file [/Users/Yanbin/Workspaces/github/spring-generic-demo/target/classes/cc/unmi/NumberService.class] - stringService: defined in file [/Users/Yanbin/Workspaces/github/spring-generic-demo/target/classes/cc/unmi/StringService.class] Action: Consider marking one of the beans as @Primary, updating the consumer to accept multiple beans, or using @Qualifier to identify the bean that should be consumed
這和普通屬性的注入時有多個可選實例時是一樣的錯誤。
總結(jié)一下
如果 Spring 在初始化泛型類時,未提供任何具體類型則會采用最上限的類型來初始化實例
@Named class Processor<T> -> new Processor<Object>()@Named class Processor<T extends Number> -> new Processor<Number>();
如果泛型類型與被注入的屬性的具體類型有關(guān)聯(lián),則由屬性類型推斷出主類型
@Named class Processor<T> {
@Inject Service<T> service;
}
此時 Spring 容器中存在 class StringService implements Service<String> 的實例,則會由屬性 service(StringService 實例) 推斷出 Processor 的具體類型是 Processor<String>
當(dāng)然這個 Processor 類也是可以定義的稍復(fù)雜一些,如
@Named class Processor<T extends Number> {
@Inject Service<T> service;
}
關(guān)于本文的示例代碼可參考 https://github.com/yabqiu/spring-generic-demo, 請運(yùn)行 mvn spring-boot:run 查看輸出結(jié)果來理解 Spring 怎么去初始化泛型類實例的。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家學(xué)習(xí)或者使用Spring能帶來一定的幫助,如果有疑問大家可以留言交流。
相關(guān)文章
從dubbo zookeeper注冊地址提取出zookeeper地址的方法
今天小編就為大家分享一篇關(guān)于從dubbo zookeeper注冊地址提取出zookeeper地址的方法,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-12-12
解決swaggerUI頁面沒有顯示Controller方法的坑
這篇文章主要介紹了解決swaggerUI頁面沒有顯示Controller方法的坑,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-06-06
SpringBoot 多線程事務(wù)回滾的實現(xiàn)
本文是基于springboot的@Async注解開啟多線程,并通過自定義注解和AOP實現(xiàn)的多線程事務(wù),避免繁瑣的手動提交/回滾事務(wù),感興趣的可以了解一下2024-02-02
新版IDEA使用Spring Initializr創(chuàng)建工程的兩種方法
這篇文章主要介紹了新版IDEA使用Spring Initializr創(chuàng)建工程(兩種方法,官方工具和IDEA),文中通過代碼示例和圖文結(jié)合的方式給大家講解的非常詳細(xì),具有一定的參考價值,需要的朋友可以參考下2024-10-10

