摘要:如果我們使用注入,如上所述,便通過調用方法注入依賴項,這一點尤其重要,因為在調用對象的構造函數時這些依賴項并不可用。
如果你看到了注解,那么一定有什么代碼在什么地方處理了它.
Alan Hohn
我教Java課程時強調的一點是注解是惰性的。換句話說,它們只是標記,可能具有某些屬性,但沒有自己的行為。因此,每當你在一段Java代碼上看到一個注解時,就意味著必須有一些其他的Java代碼來尋找那個注解并包含真正的智能來做一些有用的東西。
不幸的是,這種推理的問題在于,確切地確定哪一段代碼正在處理注解是非常困難的,特別是如果它在庫中。處理注解的代碼可能會令人困惑,因為它使用反射并且必須以非常抽象的方式編寫。所以我認為值得看看一個做得很好的例子來看看它是如何運行的。
我們詳細研究一下 Spring 框架中的 InitDestroyAnnotationBeanPostProcessor 類是如何工作的。選擇這個,因為它相對簡單,只做了一些相對容易解釋的事情, 碰巧和我手頭的工作相關。
Spring Bean 的后處理首先,我想首先解釋一下 Spring 的用途。Spring 框架所做的一件事就是“依賴注入”。這改變了我們以往用代碼將模塊串在一起的方式。例如,假設我們編寫了一些需要連接數據庫的應用程序邏輯, 但并想將提供該連接的特定硬類編碼到應用程序邏輯中,我們可以在構造函數或setter方法中將其表示為依賴項:
class MyApplication { private DataConnection data; ... public void setData(DataConnection data) { this.data = data; } ... }
當然,如果想的話, 我們可以自己編寫一個簡單的庫完成這種依賴注入,從而避免添加對 Spring 的依賴項。但是如果我們在編寫一個復雜的應用程序, 想將各模塊連接在一起,那么Spring可以非常方便。
既然沒有什么神秘的,如果我們要讓 Spring 為我們注入這些依賴,那么就會有一個權衡。Spring 需要“知道”依賴關系以及應用程序中的類和對象。Spring 處理這個問題的方法多是由 Spring 框架對對象進行實例化; 從而可以在稱為"應用程序上下文"的大數據結構中跟蹤管理這此對象。
后處理和初始化而且這里是 InitDestroyBeanPostProcessor 進入的地方 。如果 Spring 要處理實例化,那么在對象實例化完成之后,但是在應用程序開始真正的運行之前,需要進行一些“額外工作”。需要做的一件“額外工作”就是調用對象來告訴他們什么時候完全設置好,這樣他們就可以進行任何需要的額外初始化。如果我們使用“setter”注入,如上所述,便通過調用setXxx() 方法注入依賴項,這一點尤其重要,因為在調用對象的構造函數時這些依賴項并不可用。所以 Spring 需要允許用戶指定在初始化對象后才應該調用的某個方法的名稱。
Spring 一直支持使用XML配置文件來定義由 Spring 來實例化的對象,在這種情況下,有一個 "init-method" 屬性可以用來指定初始化的方法。顯然,在這種情況下,它仍然需要反射來實際查找并調用該方法。自Java 5起, 增加了注解,所以Spring 也支持帶注解的標記方法,將它們標識為Spring應該實例化的對象,識別需要注入的依賴項,以及識別應該調用的初始化和銷毀??方法。
最后一項 InitDestroyBeanPostProcessor 由其子類或其中一個子類處理。后處理器是一種特殊的對象,由Spring實例化,實現后處理器接口。因為它實現了這個接口,所以Spring會在每個Spring實例化的對象上調用一個方法,允許它修改甚至替換該對象。這是Spring采用模塊化架構方法的一部分,可以更輕松地擴展功能。
這是怎么運作的?事實上, JSR-250 確定了一些“常見”注解,包括 @PostConstruct, 用于標記初始化方法,@PreDestroy 注解, 用于注解銷毀方法的。不同的是,InitDestroyBeanPostProcessor 被設計成可以處理任何注解集,因此它提供了識別注解的方法:
public void setInitAnnotationType(Class extends Annotation> initAnnotationType) { this.initAnnotationType = initAnnotationType; } ... public void setDestroyAnnotationType(Class extends Annotation> destroyAnnotationType) { this.destroyAnnotationType = destroyAnnotationType; }
請注意,這些是普通的 setter 方法,因此這個對象本身可以使用 Spring 進行設置。就我而言,我使用Spring 的 StaticApplicationContext,見我以前的文章。
一旦 Spring 實例化了各種對象并注入了所有依賴項,它就會在所有后處理器上為每個對象調用 postProcessBeforeInitialization 方法 。這使后處理器有機會在初始化之前修改或替換對象。因為已經注入了依賴項,所以這是 InitDestroyAnnotationBeanPostProcessor 調用初始化方法的地方。
LifecycleMetadata metadata = findLifecycleMetadata(bean.getClass()); try { metadata.invokeInitMethods(bean, beanName); }
由于我們對代碼如何處理注解感興趣,我們感興趣 findLifecycleMetadata() 方法,因為這是對類進行檢查的地方。該方法檢查緩存,該緩存用于避免執行超過必要的反射,因為它可能很昂貴。如果尚未檢查該類,則調用 buildLifecycleMetadata() 方法。該方法的內容如下:
ReflectionUtils.doWithLocalMethods(targetClass, new ReflectionUtils.MethodCallback() { @Override public void doWith(Method method) throws IllegalArgumentException, IllegalAccessException { if (initAnnotationType != null) { if (method.getAnnotation(initAnnotationType) != null) { LifecycleElement element = new LifecycleElement(method); currInitMethods.add(element); } } ... } });
這里 ReflectionUtils 是一個方便的類,簡化了反射的使用。除此之外,它還將經過反射的眾多已檢查異常轉換為未經檢查的異常(?),從而使事情變得更容易。此特定方法僅迭代本地方法(即不是繼承的方法),并為每個方法調用回調。
完成所有設置之后,檢查注解的部分非常無聊; 它只是調用Java反射方法來檢查注解,如果找到它,則將該方法存儲為初始化方法。
總結事實上,這里最終發生的事情很簡單,這就是我在教反射時所要做的事情。調試使用注解來控制行為的代碼可能具有挑戰性,因為從外部來看它非常不透明,所以很難想象發生了什么(或者沒有發生)和什么時候發生。但最終,正在發生的事情只是Java代碼; 它可能不會立即顯現出代碼的位置,但它就在那里。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/72129.html
摘要:面試,是跳槽后第一個需要面對的問題而且不同公司面試的著重點不同但是卻有一個共同點基礎是必考的。對自動災難恢復有要求的表。 貌似這一點適應的行業最廣,但是我可以很肯定的說:當你從事Java一年后,重新找工作時,才會真實的感受到這句話。 工作第一年,往往是什么都充滿新鮮感,什么都學習,沖勁十足的一年;WEB行業知識更新特別快,今天一個框架的新版本,明天又是另一個新框架,有時往往根據項目的需...
摘要:面試,是跳槽后第一個需要面對的問題而且不同公司面試的著重點不同但是卻有一個共同點基礎是必考的。對自動災難恢復有要求的表。 貌似這一點適應的行業最廣,但是我可以很肯定的說:當你從事Java一年后,重新找工作時,才會真實的感受到這句話。 工作第一年,往往是什么都充滿新鮮感,什么都學習,沖勁十足的一年;WEB行業知識更新特別快,今天一個框架的新版本,明天又是另一個新框架,有時往往根據項目的需...
摘要:解釋對象關系映射集成模塊。框架中的單例是線程安全的嗎解釋框架中的生命周期。什么是織入應用的不同點解釋基于方式的切面實現。 前言 想必各位程序員已經開始準備金九銀十的秋招了,創建這個這個系列文章的目的就是為了幫助大家解決面試的問題,系列文章將會一直更新,大家如果覺得不錯可以關注我并轉發,讓更多程序兄弟看到~接下來我們進入正文環節(面試題+答案領取方式見個人主頁) 基礎篇 Spring 概...
閱讀 2262·2021-11-23 09:51
閱讀 1054·2021-11-18 10:02
閱讀 3457·2021-10-13 09:49
閱讀 1282·2021-09-22 14:57
閱讀 10564·2021-08-18 10:20
閱讀 1196·2019-08-30 15:55
閱讀 2243·2019-08-29 16:06
閱讀 3247·2019-08-29 11:14