理解Java註解及Spring的@Autowired是如何實現的
首先我們可以自己寫一個註解:
@Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface AnnoSample { String value(); }
註解使用 @interface
來標識。這個註解定義瞭一個屬性value,隻能作用於方法上,生命周期是運行時。
@Target用於指定可以放置註解的位置,這裡指定的METHOD說明該註解隻能放置到方法上面,還可以指定TYPE(類、接口、枚舉類),FIELD實例,PARAMETER形參,CONSTRUCTOR構造器等等
@Retention用於定義註解的生命周期:SOURCE是編譯期間丟棄。編譯完成後,這些註釋沒有任何意義。CLASS類加載期間丟棄,這是默認值。RUNTIME不會丟棄,可以在運行時使用反射去獲取
那麼我們就使用該註解:
public class HelloWorld { @AnnoSample(value = "hello") public void hello(){ System.out.println("hello,world"); } }
到此,創建一個註解並使用它我們已經完成瞭。
但是我們可以發現,該註解並沒有帶來任何的改變,有這個註解和沒有這個註解區別不大。那麼,我們需要知道,註解本身隻能是被看作元數據,它不包含任何業務邏輯。註解更像是一個標簽,一個聲明,表面被註釋的這個地方,將具有某種特定的邏輯。
註解讓這個方法有瞭一個標簽,讓我們知道應該去這個地方加一點邏輯。那麼怎麼去獲取這個標簽呢?
可以使用反射。
- 利用反射機制獲取一個類的 Class 對象 通過這個 class 對象可以去獲取他的每一個方法
- method,或字段 Field 等等Method,Field 等類提供瞭類似於 getAnnotation() 的方法來獲取這個字段或者方法的所有註解
- 拿到註解之後,我們可以判斷這個註解是否是我們要實現的註解,如果是則實現註解邏輯
具體實現如下:
public class Main { public static void main(String[] args) throws Exception { Class c=Class.forName("HelloWorld"); Method[] methods = c.getMethods(); for (Method method : methods) { Annotation[] annotations = method.getDeclaredAnnotations(); for (Annotation annotation : annotations) { if (annotation.annotationType()==AnnoSample.class) { System.out.println(((AnnoSample)annotation).value()); } } } } }
上面代碼就是,通過反射獲得前面所寫的HelloWorld
類的Method數組並且遍歷,並且遍歷每個方法上的所有註解,如果找到我們需要判斷的註解if (annotation.annotationType()==AnnoSample.class)
那麼就做一些邏輯處理,這裡是打印出value的值
既然已經瞭解瞭註解的基礎知識,那麼我們去看看Spring的@Autowired是怎麼實現的
@Autowired
看下源碼:
@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface Autowired { /** * Declares whether the annotated dependency is required. * <p>Defaults to {@code true}. */ boolean required() default true; }
解讀一下,該註解可以用在構造器、實例方法、形參、實例變量、註解上,生命周期是運行時。這裡的 @Documented隻是表明是否在Java doc中添加註解。
可以知道,@Autowired註解本身並沒有什麼特別的,重要的是應該是關於這個註解的特定邏輯。
邏輯所在的類,就在源碼上面有提示瞭:
連續兩次使用 Shift
進行全局搜索查詢這個類。
其中的buildAutowiringMetadata()
方法是邏輯所在:
第一個箭頭是得到當前的class,然後第二個箭頭就是去判斷 targetClass
中的所有filed,查看是否有@Autowired。 下面的doWithLocalMethods
和這裡判斷 filed類似。
通過瞭@Autowired判斷之後,執行如下
currElements.add(new AutowiredFieldElement(field, required));
這是將該字段放入一個容器中去,因為使用瞭 @Autowired的實例變量/方法不止一個,所以全部找出之後進行判斷。
在該方法的最後:
返回的是這個類和使用瞭@Autowired註解的實例集合。返回的是這個,那麼下一步應該就是對其進行註入瞭。
註入的邏輯在postProcessProperties()
方法中:
可以看到這個方法中的第一個就是調用 findAutowiringMetadata()方法,然後這個方法裡面又調用瞭我們前面分析的buildAutowiringMetadata()
,也就是說我們得到瞭類及其註解信息,然後開始調用下面的inject()
方法進行註入
可以看到,對於字段,那麼就調用反射類Field的set()方法設置值
field.set(target, getResourceToInject(target, requestingBeanName));
對於方法,就使用invoke去帶入具體的參數值進行執行:
method.invoke(target, getResourceToInject(target, requestingBeanName));
getResourceToInject()
方法的參數就是要註入的 bean 的名字,這個方法的功能就是根據這個 bean 的名字去拿到它。
到此這篇關於理解Java註解及Spring的@Autowired是如何實現的的文章就介紹到這瞭,更多相關java註解Spring的@Autowired內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Spring中的註解@Autowired實現過程全解(@Autowired 背後的故事)
- 深入分析@Resource和@Autowired註解區別
- Java @Autowired註解底層原理詳細分析
- Spring詳細講解@Autowired註解
- Java元註解Retention代碼示例介紹