91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java注解及Spring的@Autowired是如何實現的

發布時間:2021-07-10 13:35:15 來源:億速云 閱讀:194 作者:chen 欄目:開發技術

這篇文章主要介紹“Java注解及Spring的@Autowired是如何實現的”,在日常操作中,相信很多人在Java注解及Spring的@Autowired是如何實現的問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”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注解本身并沒有什么特別的,重要的是應該是關于這個注解的特定邏輯。
邏輯所在的類,就在源碼上面有提示了:

Java注解及Spring的@Autowired是如何實現的

連續兩次使用 Shift進行全局搜索查詢這個類。

其中的buildAutowiringMetadata()方法是邏輯所在:

Java注解及Spring的@Autowired是如何實現的

第一個箭頭是得到當前的class,然后第二個箭頭就是去判斷 targetClass中的所有filed,查看是否有@Autowired。 下面的doWithLocalMethods和這里判斷 filed類似。
通過了@Autowired判斷之后,執行如下

currElements.add(new AutowiredFieldElement(field, required));

這是將該字段放入一個容器中去,因為使用了 @Autowired的實例變量/方法不止一個,所以全部找出之后進行判斷。

在該方法的最后:

Java注解及Spring的@Autowired是如何實現的

返回的是這個類和使用了@Autowired注解的實例集合。返回的是這個,那么下一步應該就是對其進行注入了。

注入的邏輯在postProcessProperties()方法中:

Java注解及Spring的@Autowired是如何實現的

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

Java注解及Spring的@Autowired是如何實現的

可以看到,對于字段,那么就調用反射類Field的set()方法設置值

field.set(target, getResourceToInject(target, requestingBeanName));

對于方法,就使用invoke去帶入具體的參數值進行執行:

method.invoke(target, getResourceToInject(target, requestingBeanName));

getResourceToInject()方法的參數就是要注入的 bean 的名字,這個方法的功能就是根據這個 bean 的名字去拿到它。

到此,關于“Java注解及Spring的@Autowired是如何實現的”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

博爱县| 三门县| 萨迦县| 富锦市| 雅江县| 惠东县| 孝感市| 盐山县| 青川县| 兴和县| 定边县| 淳安县| 壤塘县| 辽阳县| 舞阳县| 冕宁县| 临洮县| 南投市| 襄垣县| 清水河县| 定远县| 竹山县| 汨罗市| 新野县| 宣城市| 遂昌县| 台中县| 五家渠市| 杂多县| 邻水| 茶陵县| 潜江市| 临江市| 龙里县| 通化市| 本溪| 湘潭市| 平罗县| 天门市| 疏勒县| 永康市|