在Spring中,我們可以通過 @Autowired注解的方式為一個方法中注入參數,那么這種方法背后到底發生了什么呢,這篇文章將講述如何用Java的注解和反射實現一個“低配版”的依賴注入。
下面是我們要做的一些事情:
- 通過 @interface的方式定義一個注解
- 為某個希望杯被注入的方法添加這個注解
- 編寫測試代碼,通過反射獲取添加了注解的方法對應的Method對象,將該方法對象設置為可訪問的,通過反射創建對象並調用這個方法,同時注入依賴數據
如上所述,我們分為三個步驟, 去加工出這個低配版的依賴注入,下面就來講講每一步的詳細步驟
我們要編寫的代碼的結構分為三部分:
- Autowired: 聲明的注解
- Demo類:含有被依賴注入的方法setStr
- Test類:通過反射獲取被Autowired注解的方法,並進行依賴注入
一:定義注解
Autowired
@Retention(RetentionPolicy.RUNTIME) public @interface Autowired {
}
首先我們通過 @interface的方式定義的一個注解, 由此也可以看出注解的地位和類,接口類似,是一種同一級的關系
@Retention是元注解,故名思義,它是用來注解(動詞)注解(名詞)的注解!(名詞),RetentionPolicy.RUNTIME 表示會將這個注解保留到運行時,這樣的話我們就能通過反射去處理注解了。
二. 為被注入的方法添加注解
下面我們為setStr方法添加一個注解
public class Demo { private String str; @Autowired public void setStr (String str) { this.str = str; } public String getStr () { return str; } }
三. 通過反射處理注解
通過反射的方式獲取並處理被注解的方法,將該方法對象設置為可訪問的,通過反射創建對象並調用這個方法,同時注入依賴數據
由於涉及到大量關於反射的API,所以對於反射機制話可以看看我以前寫的這篇文章: https://www.cnblogs.com/penghuwan/p/7580145.html
在這一步驟我們要做的事情:
- 調用Class.forName方法,傳入某個類的路徑字符串為參數,獲取該類的Class對象
- 通過調用該類Class對象的getDeclaredMethods方法,獲得聲明方法對應的Methods對象組成的數組
- 遍歷2中的Methods數組,通過調用Method對象的isAnnotationPresent方法判斷該方法有沒有加上Autowired注解,並對其中加上Autowired注解的方法做以下處理
- 通過調用Method對象的setAccessible(true);方法將對象設置為可訪問的,不這么搞下一步調用方法會出錯
- 通過Class對象的newInstance方法創建對象實例,假設其為object,則再通過method.invoke(object, “傳入的數據")調用對象的方法,注入依賴數據
- 將5中的對象實例object返回, 我們就獲得了被注入了依賴數據的對象實例了
代碼如下:
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中聲明方法對應的Method對象 Method [] methods = demoClass.getDeclaredMethods(); for (Method method : methods) { // 判斷方法是否被加上了@Autowired這個注解 if(method.isAnnotationPresent(Autowired.class)) { // 將方法設置為可調用的 method.setAccessible(true); try { Object object = demoClass.newInstance(); // 調用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()); } }
輸出結果:
我是被注入的文本
到此為止, 我們就完成了這個低配版的依賴注入了