Spring實(shí)現(xiàn)泛型注入的示例詳解
1.Spring泛型注入
創(chuàng)建一個(gè)抽象泛型類BaseDao,有參數(shù)化類型T
public abstract class BaseDao<T> {
public abstract void eat();
}每種動(dòng)物有不同的行為,貓、狗
public class Cat {
}
public class Dog {
}分別繼承BaseDao實(shí)現(xiàn)不同的行為
@Repository
public class CatDao extends BaseDao<Cat> {
@Override
public void eat() {
System.out.println("cat eat....");
}
}@Repository
public class DogDao extends BaseDao<Dog> {
@Override
public void eat() {
System.out.println("dog eat....");
}
}接著創(chuàng)建一個(gè)抽象業(yè)務(wù)類,也有參數(shù)化類型T。
注意:此處不能使用@Resource注入,會(huì)找到多個(gè)BaseDao類型的bean,無(wú)法確認(rèn)注入哪一個(gè)bean會(huì)報(bào)錯(cuò)
需要使用@Autowired注入,它有根據(jù)泛型參數(shù)匹配的邏輯,會(huì)一個(gè)個(gè)去匹配
public abstract class BaseService<T> {
@Autowired
private BaseDao<T> baseDao;
protected void eat() {
baseDao.eat();
}
}子類繼承BaseService指定參數(shù)化類型實(shí)現(xiàn)注入相應(yīng)的BaseDao
@Service
public class CatService extends BaseService<Cat> {
}@Service
public class DogService extends BaseService<Dog> {
}代碼目錄

測(cè)試
@Configuration
@ComponentScan(value = "com.monian.test.generic")
public class GenericConfig {
}public class GenericTests {
public static void main(String[] args) {
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
applicationContext.register(GenericConfig.class);
applicationContext.refresh();
DogService dogService = applicationContext.getBean(DogService.class);
CatService catService = applicationContext.getBean(CatService.class);
dogService.eat();
catService.eat();
}
}結(jié)果輸出,成功注入相應(yīng)的BaseDao

2. 關(guān)于java泛型有四種Type
GenericArrayType泛型數(shù)組類型
public class GenericArrayTypeTest {
public static class TestClass<S> {
// 泛型數(shù)組
private List<String>[] lists;
// 泛型數(shù)組
private S[] ss;
private String[] s;
}
public static void main(String[] args) {
Class cl = TestClass.class;
for (Field field : cl.getDeclaredFields()) {
System.out.println("filed's name:" + field.getName());
Type genericType = field.getGenericType();
System.out.println("Is it genericArrayType:" + (genericType instanceof GenericArrayType));
if (genericType instanceof GenericArrayType) {
GenericArrayType genericArrayType = (GenericArrayType) genericType;
System.out.println(genericArrayType.getTypeName());
Type genericComponentType = genericArrayType.getGenericComponentType();
System.out.println(genericComponentType);
}
System.out.println();
}
}
}result:
filed's name:lists
Is it genericArrayType:true
java.util.List<java.lang.String>[]
java.util.List<java.lang.String>
filed's name:ss
Is it genericArrayType:true
S[]
S
filed's name:s
Is it genericArrayType:false
ParameterizedType參數(shù)化類型
public class ParameterizedTypeTest {
public static abstract class Test<T> {
public abstract void test(T t);
}
public static class TestClass<T extends Number> extends Test<Integer> {
private List<T> tList;
private List<? extends Number> list;
private Map<String, Integer> map;
@Override
public void test(Integer o) {
}
}
public static void main(String[] args) {
TestClass<Integer> tt = new TestClass<>();
Class cl = tt.getClass();
Type genericSuperclass = cl.getGenericSuperclass();
assert genericSuperclass instanceof ParameterizedType;
ParameterizedType parameterizedType1 = (ParameterizedType) genericSuperclass;
System.out.println(parameterizedType1.getActualTypeArguments()[0]);
for (Field field : cl.getDeclaredFields()) {
System.out.println("field's name:" + field.getName());
Type genericType = field.getGenericType();
System.out.println("Is it ParameterizedType:" + (genericType instanceof ParameterizedType));
if (genericType instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) genericType;
for (Type type : parameterizedType.getActualTypeArguments()) {
System.out.println("actualType:" + type);
System.out.println(type.getTypeName());
}
}
System.out.println();
}
}
}result:
class java.lang.Integer
field's name:tList
Is it ParameterizedType:true
actualType:T
T
field's name:list
Is it ParameterizedType:true
actualType:? extends java.lang.Number
? extends java.lang.Number
field's name:map
Is it ParameterizedType:true
actualType:class java.lang.String
java.lang.String
actualType:class java.lang.Integer
java.lang.Integer
TypeVariable 類型變量
public class TypeVariableTest {
public static class TestClass<S extends Number, T> {
private Map<S, T> map;
}
public static void main(String[] args) throws Exception {
Class cl = TestClass.class;
Field field = cl.getDeclaredField("map");
Type genericType = field.getGenericType();
ParameterizedType parameterizedType = (ParameterizedType) genericType;
for (Type type : parameterizedType.getActualTypeArguments()) {
TypeVariable typeVariable = (TypeVariable) type;
// 類型變量名
System.out.println(typeVariable.getName());
// 變量上邊界
Type[] bounds = typeVariable.getBounds();
System.out.println(Arrays.toString(bounds));
}
}
}result:
S
[class java.lang.Number]
T
[class java.lang.Object]
WildcardType 通配符類型
public class WildcardTypeTest {
public static class TestClass {
private List<? extends Number> lists;
private Set<?> sets;
private Map<? extends Number, ? super String> map;
}
public static void main(String[] args) {
Class cl = TestClass.class;
for (Field field : cl.getDeclaredFields()) {
System.out.println("filed's name:" + field.getName());
Type genericType = field.getGenericType();
ParameterizedType parameterizedType = (ParameterizedType) genericType;
for (Type type : parameterizedType.getActualTypeArguments()) {
// 通配符類型
WildcardType wildcardType = (WildcardType) type;
System.out.println(wildcardType.getTypeName());
// 上邊界
System.out.println("上邊界" + Arrays.toString(wildcardType.getUpperBounds()));
// 下邊界
System.out.println("下邊界" + Arrays.toString(wildcardType.getLowerBounds()));
}
System.out.println();
}
}
}
result:
filed's name:lists
? extends java.lang.Number
上邊界[class java.lang.Number]
下邊界[]
filed's name:sets
?
上邊界[class java.lang.Object]
下邊界[]
filed's name:map
? extends java.lang.Number
上邊界[class java.lang.Number]
下邊界[]
? super java.lang.String
上邊界[class java.lang.Object]
下邊界[class java.lang.String]
注:spring對(duì)泛型的解析主要是在ResolvableType類,掌握上述的基本知識(shí)后可以去閱讀下相關(guān)源碼
到此這篇關(guān)于Spring實(shí)現(xiàn)泛型注入的示例詳解的文章就介紹到這了,更多相關(guān)Spring泛型注入內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot實(shí)現(xiàn)連接nacos并支持多環(huán)境部署
這篇文章主要介紹了SpringBoot實(shí)現(xiàn)連接nacos并支持多環(huán)境部署方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06
如何通過(guò)Java監(jiān)聽MySQL數(shù)據(jù)的變化
對(duì)于二次開發(fā)來(lái)說(shuō),很大一部分就找找文件和找數(shù)據(jù)庫(kù)的變化情況,下面這篇文章主要給大家介紹了關(guān)于如何通過(guò)Java監(jiān)聽MySQL數(shù)據(jù)的變化的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-03-03
用java實(shí)現(xiàn)在txt文本中寫數(shù)據(jù)和讀數(shù)據(jù)的方法
今天小編就為大家分享一篇用java實(shí)現(xiàn)在txt文本中寫數(shù)據(jù)和讀數(shù)據(jù)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-07-07
SpringSecurity?Web權(quán)限方案實(shí)現(xiàn)全過(guò)程
Spring Security是一個(gè)功能強(qiáng)大且高度可定制的身份驗(yàn)證和授權(quán)框架,專門用于保護(hù)Java應(yīng)用程序的Web集成,下面這篇文章主要給大家介紹了關(guān)于SpringSecurity?Web權(quán)限方案實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2024-01-01
SpringBoot集成JPA持久層框架,簡(jiǎn)化數(shù)據(jù)庫(kù)操作
JPA(Java Persistence API)意即Java持久化API,是Sun官方在JDK5.0后提出的Java持久化規(guī)范。主要是為了簡(jiǎn)化持久層開發(fā)以及整合ORM技術(shù),結(jié)束Hibernate、TopLink、JDO等ORM框架各自為營(yíng)的局面。JPA是在吸收現(xiàn)有ORM框架的基礎(chǔ)上發(fā)展而來(lái),易于使用,伸縮性強(qiáng)。2021-06-06
maven多模塊項(xiàng)目依賴管理與依賴?yán)^承詳解
這篇文章主要介紹了maven多模塊項(xiàng)目依賴管理與依賴?yán)^承詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12

