springaop的源碼解析
① 「Spring 」「AOP 容器」不看源碼就帶你認識核心流程以及運作原理
前一篇文章主要介紹了 spring 核心特性機制的 IOC 容器機制和核心運作原理,接下來我們去介紹另外一個較為核心的功能,那就是 AOP 容器機制,主要負責承接前一篇代理模式機制中動態代理:JDKProxy 和 CglibProxy 的功能機制之後,我們開始研究一下如何實現一下相關的 AOP 容器代理機制的。
實現的基本實現原理就是後置處理器:BeanPostProcessor 機制,實現動態化植入機制。
bean 在初始化的時候會進行調用對應的 BeanPostProcessor 的對應的方法會進行織入。
主要取決於 wrapIfNecessary 方法:
如果是基礎設施類型,則直接回進行返回該 bean 對象,不會進行相關的初始化對應的 aspectj 的動態織入機制。
會進行尋找相關的 Bean 對應的何時的加強通知類。
則會對該 bean 對象,額外進行增強操作生成相關的代理對象,並返回該執行之後的對象,否則會直接返回該對象即可。
getAdvicesAndAdvisorsForBean 方法是我們篩選 Advice 增強類的核心方法,主要用於過濾和篩選對應該 bean 的何時的增強器數組信息。
主要用於調用 的**findCandidateAdvisors()**方法,其內部會進行先關的核心構建相關的 Aspectj 的類的相關實現操作
advisorsFactory.getAdvisors 獲取通知器
切點類處理操作到此為止,還不完整接下來才是構建動態代理對象的真正執行操作,
擴展相關的篩選出的通知器列表,extendAdvisors 方法,通知器列表首部添加一個 DefaultPointcutAposr 類型的通知器,也就是 ExposeInvocationInterceptor.ADVISOR 的實現機制。
proxy-target-class 的屬性值,代表是否可以支持代理實現類,默認採用的 false 代表著,當 bean 有實現介面的時候,會直接採用 jdk 的動態代理機制生成代理對象,如果是 true,則代表著使用 cglib 進行生成代理對象。
復制代碼
前提是必須要配置相關的 expose-proxy 屬性配置值為 true,才會進行暴露對應的代理機制。
為了解決目標方法調用同對象中的其他方法,其他方法的切面邏輯是無法實現,因為會涉及到相關的 this 操作而不是 proxy 對象機制。
可以實現使用 AopContext.currentProxy()強制轉換為當前的代理對象。
獲取相關的對應方法的攔截器棧鏈路,如果沒有獲取到相關的緩存鏈路,則會直接調用相關的 獲取先關的攔截器鏈。
會進行先關的 PointcutAdvisor 類型通知器,這里會調用相關的通知器所持有的切點(Pointcut)對類和方法進行匹配,匹配沖過這說明相關的向當前的方法進行織入邏輯控制。此外還會通過 geIntercptors()方法對非 MethodIntercptor 類型的通知進行轉換。返回相關的攔截器數組,並且隨後存入緩存中。
則會直接通過代理機制的反射控制進行調用執行即可。
則例如 jdkDynamicAutoProxy 對象進行調用構建 ReflectiveMethodInvocation 對象,例如它的 process 方法啟動攔截器棧的 invoke 方法。
處理返回值,並且返回該值。
② 怎麼閱讀Spring源碼
學習源碼是一件非常耗時費力的事情,需要有足夠的時間和持久的耐心,下面是我閱讀郝佳老師的《Spring源碼深度解析》所做的記錄,書中以Spring3.2講解,使用jdk1.7。
准備工作
1. 安裝github:現在spring源代碼都在github管理,所以首先需要下載githup,下;
2. 安裝gradle構建工具: 下載完後進行解壓到任意盤符,然後增加環境變數GRADLE_HOME,並在環境變數bin中增加%GRADLE_HOME%/bin,打開DOS窗口,運行gradle -v,出現版本號等信息,表示安裝成功;
3. 下載Spring源碼:首先打開git shell,切換到你的工作目錄,然後輸入以下命令:git clone git://github.com/SpringSource/Spring-framework.git,後面一串是源碼下載地址。大概半小時的樣子,就可以下載完成,這時候在你的工作目錄中就會出現Spring-framework的目錄,裡面有Spring各組件的源碼包;
4. 構建導入:下載下來的代碼不能直接導入Eclipse,要先轉換成Eclipse能讀取的形式。因為所有組件都會依賴spring-core,所有我們首先要轉換Spring-core工程,在命令窗口切換到Spring-core工程,運行gradle cleanidea eclipse命令,我們會看到開始下載工程所依賴的jar包,幾分鍾後執行完畢,再來看Spring-core文件夾,多了.classpath、.project等文件,這是Eclipse工程所必須的,然後可以把他導入到eclipse。因為大部分Spring組件都會用到 spring-beans、spring-context、spring-aop,而他們又依賴spring-expression、spring-instrument,所以我們乾脆先把這些工程都進行轉換並導入eclipse。
③ 如何查看spring源碼
1.准備工作:在官網上下載了Spring源代碼之後,導入Eclipse,以方便查詢。
2.打開我們使用Spring的項目工程,找到Web.xml這個網站系統配置文件,在其中找到Spring的初始化信息:
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
由配置信息可知,我們開始的入口就這里ContextLoaderListener這個監聽器。
在源代碼中我們找到了這個類,它的定義是:
public class ContextLoaderListener extends ContextLoader
implements ServletContextListener {
…
/**
* Initialize the root web application context.
*/
public void contextInitialized(ServletContextEvent event) {
this.contextLoader = createContextLoader();
if (this.contextLoader == null) {
this.contextLoader = this;
}
this.contextLoader.initWebApplicationContext(event.getServletContext());
}
...
}
該類繼續了ContextLoader並實現了監聽器,關於Spring的信息載入配置、初始化便是從這里開始了,具體其他閱讀另外寫文章來深入了解。
二、關於IOC和AOP
關於Spring IOC 網上很多相關的文章可以閱讀,那麼我們從中了解到的知識點是什麼?
1)IOC容器和AOP切面依賴注入是Spring是核心。
IOC容器為開發者管理對象之間的依賴關系提供了便利和基礎服務,其中Bean工廠(BeanFactory)和上下文(ApplicationContext)就是IOC的表現形式。BeanFactory是個介面類,只是對容器提供的最基本服務提供了定義,而DefaultListTableBeanFactory、XmlBeanFactory、ApplicationContext等都是具體的實現。
介面:
public interface BeanFactory {
//這里是對工廠Bean的轉義定義,因為如果使用bean的名字檢索IOC容器得到的對象是工廠Bean生成的對象,
//如果需要得到工廠Bean本身,需要使用轉義的名字來向IOC容器檢索
String FACTORY_BEAN_PREFIX = "&";
//這里根據bean的名字,在IOC容器中得到bean實例,這個IOC容器就象一個大的抽象工廠,用戶可以根據名字得到需要的bean
//在Spring中,Bean和普通的JAVA對象不同在於:
//Bean已經包含了我們在Bean定義信息中的依賴關系的處理,同時Bean是已經被放到IOC容器中進行管理了,有它自己的生命周期
Object getBean(String name) throws BeansException;
//這里根據bean的名字和Class類型來得到bean實例,和上面的方法不同在於它會拋出異常:如果根名字取得的bean實例的Class類型和需要的不同的話。
Object getBean(String name, Class requiredType) throws BeansException;
//這里提供對bean的檢索,看看是否在IOC容器有這個名字的bean
boolean containsBean(String name);
//這里根據bean名字得到bean實例,並同時判斷這個bean是不是單件,在配置的時候,默認的Bean被配置成單件形式,如果不需要單件形式,需要用戶在Bean定義信息中標注出來,這樣IOC容器在每次接受到用戶的getBean要求的時候,會生成一個新的Bean返回給客戶使用 - 這就是Prototype形式
boolean isSingleton(String name) throws NoSuchBeanDefinitionException;
//這里對得到bean實例的Class類型
Class getType(String name) throws NoSuchBeanDefinitionException;
//這里得到bean的別名,如果根據別名檢索,那麼其原名也會被檢索出來
String[] getAliases(String name);
}
實現:
XmlBeanFactory的實現是這樣的:
public class XmlBeanFactory extends DefaultListableBeanFactory {
//這里為容器定義了一個默認使用的bean定義讀取器,在Spring的使用中,Bean定義信息的讀取是容器初始化的一部分,但是在實現上是和容器的注冊以及依賴的注入是分開的,這樣可以使用靈活的 bean定義讀取機制。
private final XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(this);
//這里需要一個Resource類型的Bean定義信息,實際上的定位過程是由Resource的構建過程來完成的。
public XmlBeanFactory(Resource resource) throws BeansException {
this(resource, null);
}
//在初始化函數中使用讀取器來對資源進行讀取,得到bean定義信息。這里完成整個IOC容器對Bean定義信息的載入和注冊過程
public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws
BeansException {
super(parentBeanFactory);
this.reader.loadBeanDefinitions(resource);
}