如何用Java注解和反射實現(xiàn)依賴注入
概述
在Spring中,我們可以通過 @Autowired注解的方式為一個方法中注入?yún)?shù),那么這種方法背后到底發(fā)生了什么呢,這篇文章將講述如何用Java的注解和反射實現(xiàn)一個“低配版”的依賴注入。
下面是我們要做的一些事情:
- 通過 @interface的方式定義一個注解
- 為某個希望杯被注入的方法添加這個注解
- 編寫測試代碼,通過反射獲取添加了注解的方法對應(yīng)的Method對象,將該方法對象設(shè)置為可訪問的,通過反射創(chuàng)建對象并調(diào)用這個方法,同時注入依賴數(shù)據(jù)
如上所述,我們分為三個步驟, 去加工出這個低配版的依賴注入,下面就來講講每一步的詳細步驟
我們要編寫的代碼的結(jié)構(gòu)分為三部分:
- Autowired: 聲明的注解
- Demo類:含有被依賴注入的方法
- setStrTest類:通過反射獲取被Autowired注解的方法,并進行依賴注入
定義注解
Autowired
@Retention(RetentionPolicy.RUNTIME) public @interface Autowired { }
首先我們通過 @interface的方式定義的一個注解, 由此也可以看出注解的地位和類,接口類似,是一種同一級的關(guān)系
@Retention是元注解,故名思義,它是用來注解(動詞)注解(名詞)的注解?。~),RetentionPolicy.RUNTIME 表示會將這個注解保留到運行時,這樣的話我們就能通過反射去處理注解了。
為被注入的方法添加注解
下面我們?yōu)閟etStr方法添加一個注解
public class Demo { private String str; @Autowired public void setStr (String str) { this.str = str; } public String getStr () { return str; } }
通過反射處理注解
通過反射的方式獲取并處理被注解的方法,將該方法對象設(shè)置為可訪問的,通過反射創(chuàng)建對象并調(diào)用這個方法,同時注入依賴數(shù)據(jù)
由于涉及到大量關(guān)于反射的API,所以對于反射機制話可以看看我以前寫的這篇文章: https://www.cnblogs.com/penghuwan/p/7580145.html
在這一步驟我們要做的事情:
1.調(diào)用Class.forName方法,傳入某個類的路徑字符串為參數(shù),獲取該類的Class對象
2.通過調(diào)用該類Class對象的getDeclaredMethods方法,獲得聲明方法對應(yīng)的Methods對象組成的數(shù)組
3.遍歷2中的Methods數(shù)組,通過調(diào)用Method對象的isAnnotationPresent方法判斷該方法有沒有加上Autowired注解,并對其中加上Autowired注解的方法做以下處理
4.通過調(diào)用Method對象的setAccessible(true);方法將對象設(shè)置為可訪問的,不這么搞下一步調(diào)用方法會出錯
5.通過Class對象的newInstance方法創(chuàng)建對象實例,假設(shè)其為object,則再通過method.invoke(object, “傳入的數(shù)據(jù)")調(diào)用對象的方法,注入依賴數(shù)據(jù)
6.將5中的對象實例object返回, 我們就獲得了被注入了依賴數(shù)據(jù)的對象實例了
代碼如下:
Test.java
import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; public class Test { /** * 這個方法會將一段文本注入到某個類中添加了@Autowired注解的方法中,并將實例對象返回 */ public static Object injectStrToInstance (String ClassName,String str) throws ClassNotFoundException { // 獲取Demo的Class對象 Class demoClass = Class.forName(ClassName); // 從Class對象中獲取Demo中聲明方法對應(yīng)的Method對象 Method [] methods = demoClass.getDeclaredMethods(); for (Method method : methods) { // 判斷方法是否被加上了@Autowired這個注解 if(method.isAnnotationPresent(Autowired.class)) { // 將方法設(shè)置為可調(diào)用的 method.setAccessible(true); try { Object object = demoClass.newInstance(); // 調(diào)用method方法,向其中注入str字符串 method.invoke(object,str); return object; } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } } } return null; } public static void main (String args []) throws ClassNotFoundException { // 進行依賴注入,并取得注入后的Demo的對象實例 Demo demo1 = (Demo)injectStrToInstance("Demo", "我是被注入的文本"); // 輸出一下看看我們的文本是不是被成功注入進去了 System.out.println(demo1.getStr()); } }
輸出結(jié)果:
我是被注入的文本
到此為止, 我們就完成了這個低配版的依賴注入了
以上就是如何用Java注解和反射實現(xiàn)依賴注入的詳細內(nèi)容,更多關(guān)于Java注解和反射實現(xiàn)依賴注入的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java中的拷貝數(shù)組CopyOnWriteArrayList詳解
這篇文章主要介紹了Java中的拷貝數(shù)組CopyOnWriteArrayList詳解,ArrayList和LinkedList都不是線程安全的,如果需要線程安全的List,可以使用synchronizedList來生成一個同步list,但是這個同步list的方法都是通過synchronized修飾來保證同步的,需要的朋友可以參考下2023-12-12解決mybatis使用foreach批量insert異常的問題
這篇文章主要介紹了解決mybatis使用foreach批量insert異常的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-01-01Mybatis實現(xiàn)單個和批量定義別名typeAliases
這篇文章主要介紹了Mybatis實現(xiàn)單個和批量定義別名typeAliases,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09