理解Java注解及Spring的@Autowired是如何實(shí)現(xiàn)的
首先我們可以自己寫一個(gè)注解:
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface AnnoSample {
String value();
}
注解使用 @interface來標(biāo)識。這個(gè)注解定義了一個(gè)屬性value,只能作用于方法上,生命周期是運(yùn)行時(shí)。
@Target用于指定可以放置注解的位置,這里指定的METHOD說明該注解只能放置到方法上面,還可以指定TYPE(類、接口、枚舉類),F(xiàn)IELD實(shí)例,PARAMETER形參,CONSTRUCTOR構(gòu)造器等等
@Retention用于定義注解的生命周期:SOURCE是編譯期間丟棄。編譯完成后,這些注釋沒有任何意義。CLASS類加載期間丟棄,這是默認(rèn)值。RUNTIME不會丟棄,可以在運(yùn)行時(shí)使用反射去獲取
那么我們就使用該注解:
public class HelloWorld {
@AnnoSample(value = "hello")
public void hello(){
System.out.println("hello,world");
}
}
到此,創(chuàng)建一個(gè)注解并使用它我們已經(jīng)完成了。
但是我們可以發(fā)現(xiàn),該注解并沒有帶來任何的改變,有這個(gè)注解和沒有這個(gè)注解區(qū)別不大。那么,我們需要知道,注解本身只能是被看作元數(shù)據(jù),它不包含任何業(yè)務(wù)邏輯。注解更像是一個(gè)標(biāo)簽,一個(gè)聲明,表面被注釋的這個(gè)地方,將具有某種特定的邏輯。
注解讓這個(gè)方法有了一個(gè)標(biāo)簽,讓我們知道應(yīng)該去這個(gè)地方加一點(diǎn)邏輯。那么怎么去獲取這個(gè)標(biāo)簽?zāi)兀?br /> 可以使用反射。
- 利用反射機(jī)制獲取一個(gè)類的 Class 對象 通過這個(gè) class 對象可以去獲取他的每一個(gè)方法
- method,或字段 Field 等等Method,F(xiàn)ield 等類提供了類似于 getAnnotation() 的方法來獲取這個(gè)字段或者方法的所有注解
- 拿到注解之后,我們可以判斷這個(gè)注解是否是我們要實(shí)現(xiàn)的注解,如果是則實(shí)現(xiàn)注解邏輯
具體實(shí)現(xiàn)如下:
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數(shù)組并且遍歷,并且遍歷每個(gè)方法上的所有注解,如果找到我們需要判斷的注解if (annotation.annotationType()==AnnoSample.class)那么就做一些邏輯處理,這里是打印出value的值
既然已經(jīng)了解了注解的基礎(chǔ)知識,那么我們?nèi)タ纯碨pring的@Autowired是怎么實(shí)現(xiàn)的
@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;
}
解讀一下,該注解可以用在構(gòu)造器、實(shí)例方法、形參、實(shí)例變量、注解上,生命周期是運(yùn)行時(shí)。這里的 @Documented只是表明是否在Java doc中添加注解。
可以知道,@Autowired注解本身并沒有什么特別的,重要的是應(yīng)該是關(guān)于這個(gè)注解的特定邏輯。
邏輯所在的類,就在源碼上面有提示了:

連續(xù)兩次使用 Shift進(jìn)行全局搜索查詢這個(gè)類。
其中的buildAutowiringMetadata()方法是邏輯所在:

第一個(gè)箭頭是得到當(dāng)前的class,然后第二個(gè)箭頭就是去判斷 targetClass中的所有filed,查看是否有@Autowired。 下面的doWithLocalMethods和這里判斷 filed類似。
通過了@Autowired判斷之后,執(zhí)行如下
currElements.add(new AutowiredFieldElement(field, required));
這是將該字段放入一個(gè)容器中去,因?yàn)槭褂昧?@Autowired的實(shí)例變量/方法不止一個(gè),所以全部找出之后進(jìn)行判斷。
在該方法的最后:

返回的是這個(gè)類和使用了@Autowired注解的實(shí)例集合。返回的是這個(gè),那么下一步應(yīng)該就是對其進(jìn)行注入了。
注入的邏輯在postProcessProperties()方法中:

可以看到這個(gè)方法中的第一個(gè)就是調(diào)用 findAutowiringMetadata()方法,然后這個(gè)方法里面又調(diào)用了我們前面分析的buildAutowiringMetadata(),也就是說我們得到了類及其注解信息,然后開始調(diào)用下面的inject()方法進(jìn)行注入

可以看到,對于字段,那么就調(diào)用反射類Field的set()方法設(shè)置值
field.set(target, getResourceToInject(target, requestingBeanName));
對于方法,就使用invoke去帶入具體的參數(shù)值進(jìn)行執(zhí)行:
method.invoke(target, getResourceToInject(target, requestingBeanName));
getResourceToInject()方法的參數(shù)就是要注入的 bean 的名字,這個(gè)方法的功能就是根據(jù)這個(gè) bean 的名字去拿到它。
到此這篇關(guān)于理解Java注解及Spring的@Autowired是如何實(shí)現(xiàn)的的文章就介紹到這了,更多相關(guān)java注解Spring的@Autowired內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring框架構(gòu)造注入type屬性實(shí)例詳解
這篇文章主要介紹了Spring框架構(gòu)造注入type屬性實(shí)例詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
Java利用廣度優(yōu)先搜索實(shí)現(xiàn)抓牛問題
廣度優(yōu)先搜索是最簡便的圖的搜索算法之一,這一算法也是很多重要的圖的算法的原型。本文將利用廣度優(yōu)先搜索實(shí)現(xiàn)抓牛問題,感興趣的可以了解下2022-06-06
spring-security關(guān)閉登錄框的實(shí)現(xiàn)示例
這篇文章主要介紹了spring-security關(guān)閉登錄框的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05
詳解使用Spring的restTemplete進(jìn)行Http請求
本篇文章主要介紹了詳解使用Spring的restTemplete進(jìn)行Http請求,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
Spring?Boot條件注解之@ConditionalOnProperty完全解析
這篇文章主要介紹了SpringBoot中的@ConditionalOnProperty注解,通過配置文件屬性值控制Bean或配置類的加載,實(shí)現(xiàn)功能開關(guān)和環(huán)境配置,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2025-02-02
Eclipse配置python開發(fā)環(huán)境過程圖解
這篇文章主要介紹了Eclipse配置python開發(fā)環(huán)境過程圖解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
java如何自動(dòng)補(bǔ)齊數(shù)值至指定位數(shù)
這篇文章主要介紹了java如何自動(dòng)補(bǔ)齊數(shù)值至指定位數(shù)問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01
詳解Springboot集成sentinel實(shí)現(xiàn)接口限流入門
這篇文章主要介紹了詳解Springboot集成sentinel實(shí)現(xiàn)接口限流入門,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
本地編譯打包項(xiàng)目部署到服務(wù)器并且啟動(dòng)方式
這篇文章主要介紹了本地編譯打包項(xiàng)目部署到服務(wù)器并且啟動(dòng)方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02

