【Java】利用注解和反射實現一個"低配版"的依賴注入


在Spring中,我們可以通過 @Autowired注解的方式為一個方法中注入參數,那么這種方法背后到底發生了什么呢,這篇文章將講述如何用Java的注解和反射實現一個“低配版”的依賴注入。


下面是我們要做的一些事情:

  1. 通過 @interface的方式定義一個注解
  2. 為某個希望杯被注入的方法添加這個注解
  3. 編寫測試代碼,通過反射獲取添加了注解的方法對應的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

在這一步驟我們要做的事情:

  1. 調用Class.forName方法,傳入某個類的路徑字符串為參數,獲取該類的Class對象
  2. 通過調用該類Class對象的getDeclaredMethods方法,獲得聲明方法對應的Methods對象組成的數組
  3. 遍歷2中的Methods數組,通過調用Method對象的isAnnotationPresent方法判斷該方法有沒有加上Autowired注解,並對其中加上Autowired注解的方法做以下處理
  4. 通過調用Method對象的setAccessible(true);方法將對象設置為可訪問的,不這么搞下一步調用方法會出錯
  5. 通過Class對象的newInstance方法創建對象實例,假設其為object,則再通過method.invoke(object, “傳入的數據")調用對象的方法,注入依賴數據
  6. 將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());
    }
}

 


輸出結果:

我是被注入的文本

到此為止, 我們就完成了這個低配版的依賴注入了

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM