程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA綜合教程 >> Spring源碼解析,spring源碼

Spring源碼解析,spring源碼

編輯:JAVA綜合教程

Spring源碼解析,spring源碼


我們先來看類圖吧:

除了BeanFactory這一支的接口,AbstractBeanFactory主要實現了AliasRegistry和SingletonBeanRegistry接口.

這邊主要提供了這樣的三個功能: 別名管理,單例創建與注冊,工廠方法FactoryBean支持.

 

我們來看看這些接口,類的主要職責吧:

BeanFactory Spring IOC容器的根接口

-- HierachicalBeanFactory 實現容器的繼承,就是可以有父 BeanFactory

-- -- ConfigureabelBeanFactory 提供factory的配置功能

AliasRegistry 定義bean name的別名管理

-- SimpleAliasRegistry 在實現別名管理接口基礎上,添加一個canonicalName查找類真是名稱api

SingletonBeanRegistry 提供單例注冊,查詢服務

-- DefaultSingletonBeanRegistry 實現單例與DisposableBean的生命周期管理(創建,維護,銷毀)

-- -- FactoryBeanRegistrySupport 添加工廠方式創建類FactoryBean的支持

-- -- -- AbstractBeanFactory BeanFactory的抽象實現.

 

也就是說這邊可以分為這樣幾類職責:

a, 類別名管理

b, 單例生命周期管理

c, 工廠方法初始化類對應的FactoryBean

d, BeanFactory容器

 

BeanFactory容器的職責(BeanFactory,HierachicalBeanFactory,ConfigureableBeanFactory)在上一篇文章<Spring源碼解析 - BeanFactory接口體系解讀>裡已經分析過,有興趣可以看下.

我們今天主要分析其他的三個職責,如下的接口與類(順便做目錄):

1. AliasRegistry 定義bean name的別名管理

2. SimpleAliasRegistry 實現別名管理接口

3. SingletonBeanRegistry 提供單例注冊,查詢服務

4. DefaultSingletonBeanRegistry 實現單例與DisposableBean的生命周期管理(創建,維護,銷毀)

5. FactoryBeanRegistrySupport 添加工廠方式創建類FactoryBean的支持

 

 

1. AliasRegistry 定義bean name的別名管理

提供別名的注冊,查找,刪除,判斷定義.

 看個類圖就行,不用展開.

 

 

2. SimpleAliasRegistry 實現別名管理接口

這邊除了實現接口定義的api,還添加了兩個公共api:

  批量校驗別名public void resolveAliases(StringValueResolver valueResolver)和查找別名對應的原始類名public String canonicalName(String name)

 

這邊以別名為key緩存數據.

1     /** Map from alias to canonical name */
2     private final Map<String, String> aliasMap = new ConcurrentHashMap<String, String>(16);

 

分析下api實現邏輯吧:

2.1 別名注冊 registerAlias(String name, String alias)

2.2 刪除別名public void removeAlias(String alias)

  校驗下,如果別名不存在報錯:throw new IllegalStateException("No alias '" + alias + "' registered");

 

2.3 判斷別名是否存在

  直接使用ConcurrentHashMap的containsKey

2.4 獲取別名public String[] getAliases(String name)

  主要邏輯是加了個鎖,然後是遞歸調用retrieveAliases,查找多層次的別名(就是查找別名的別名這些下去)

2.5 使用StringValueResolver解析類名,別名後,進行循環依賴的校驗

  這邊使用的是StringValueResolver的接口,具體實現需要靠注入

 

 

3. SingletonBeanRegistry 提供單例注冊,查詢服務

這邊定義的單例注冊,有點門道,主要就是相對BeanFactory的api而言有點low,沒有做附加的處理.

  注冊的時候不管注入afterPropertiesSet的初始化回調.

  查找的時候不管還沒初始化的單例不說,還不管別名問題,不管FactoryBean如何區分是獲取FactoryBean本身還是getObject初始化的實例.

 

咱們一個個api分析吧.

3.1 注冊單例 void registerSingleton(String beanName, Object singletonObject);

  這邊的實現不會再調用 初始化回調函數,如InitializingBean 的afterPropertiesSet,所以這邊應該接收的是完成初始化的實例

  同理也不會調用銷毀的回調,如DisposableBean的destroy

  這跟標准的BeanFactory中注冊單例是明顯不同的,因為那邊是會調用各種回調.

 

3.2 查找單例 Object getSingleton(String beanName); 

      String[] getSingletonNames();

      int getSingletonCount();

  設計於訪問手動注冊的單例.

  這邊只會查找已經初始化完畢 的單例,有bean definition但沒有實例化的這邊查找不到.

  這邊也不會處理FactoryBean的情況(就是具體獲取getObject還是factoryBean本身的區分,&),別名也需要預先轉化好了來查.

 

3.3 判斷是否保護單例 boolean containsSingleton(String beanName);

  只有單例已經實例化才會返回true,剩下的看3.2 查找單例的說明吧,一樣的.

 

 

4. DefaultSingletonBeanRegistry 實現單例與DisposableBean的生命周期管理(創建,維護,銷毀)

 在SingletonBeanRegistry注冊的基礎上,添加單例的共享.

也支持容器關閉時,DisposableBean實例的銷毀

 

4.1 這邊注冊時,通過下面四個變量來維護:

  Map<String, Object> singletonObjects 緩存以bean name為key的單例實例

  Map<String, ObjectFactory> singletonFactories 緩存以bean name 為key的ObjectFactory

  Map<String, Object> earlySingletonObjects 用於解決單例時的循環依賴,這邊緩存以bean name為key的預初始化單例

  Set<String> registeredSingletons 已經注冊好的單例bean name

  這邊singletonObjects和registeredSingletons的數據應該是同步的,只是適用於不同的場景,但他們倆跟singletonFactories 和earlySingletonObjects分別互斥,就是singletonObjects裡有了,這兩個肯定沒有.

  

  同時這邊也有inCreationCheckExclusions和singletonsCurrentlyInCreation進行鎖控制的概念.

    singletonsCurrentlyInCreation緩存bean正在被初始化,這樣就不能再發起初始化;

    inCreationCheckExclusions 直接緩存當前不能加載的bean

  這部分看個例子就,清晰了,初始化前需要先使用beforeSingletonCreation判斷

    這邊inCreationCheckExclusions不包含beanName才會去判斷singletonsCurrentlyInCreation

1     protected void beforeSingletonCreation(String beanName) {
2         if (!this.inCreationCheckExclusions.containsKey(beanName) &&
3                 this.singletonsCurrentlyInCreation.put(beanName, Boolean.TRUE) != null) {
4             throw new BeanCurrentlyInCreationException(beanName);
5         }
6     }

 

4.2 管理bean的依賴問題

  使用如下三個屬性進行管理:

  Map<String, Set<String>> containedBeanMap  依賴的bean name為key , 就是依賴類 -> 查找 被依賴的類

  Map<String, Set<String>> dependentBeanMap  依賴的原始bean name為key

  Map<String, Set<String>> dependenciesForBeanMap  被依賴的bean name為key

 

4.3 bean 銷毀

  這不過跟初始化類似,自行看代碼比較簡單.    

 

 

5. FactoryBeanRegistrySupport 添加工廠方式創建類FactoryBean的支持

添加對FactoryBean的支持,就是使用工廠方法初始化類.

這裡主要涉及3個新概念:FactoryBean,BeanPostProcessor和AccessController.這三個概念懂了,源碼也就分析完了.

 

5.1 FactoryBean,通過T getObject() api提供簡單工廠方法,可用用於創建單例,原型模式的實例.主要用於創建過程復雜,xml配置不方便的情況.

  其實這個就是使用spring的接口對簡單工廠設計模式做了一個規范,方便大家在spring中配置使用.

  具體直接看<Spring配置bean的方法(工廠方法和Factorybean)>

 

5.2 BeanPostProcessor用於bean 初始化時進行功能增強,類似web開發中的filter.

  這邊有兩個api:

  postProcessBeforeInitialization 在類初始化前調用,比InitializaingBean 的 setPropertiesSet 和 xml文件中自定義的init-method方法執行都早

  postProcessAfterInitialization 類初始話後調用,在InitializaingBean 的 setPropertiesSet 和 xml文件中自定義的init-method方法之後執行

 

5.3 AccessController jdk的安全控制,跟spring關聯不大,還是度娘吧,不多寫了.

  在 Java 中將執行程序分成本地和遠程兩種,本地代碼默認視為可信任的,而遠程代碼則被看作是不受信的。對於授信的本地代碼,可以訪問一切本地資源。

  在應用開發中還有一些關於安全的復雜用法,其中最常用到的 API 就是 doPrivileged。doPrivileged 方法能夠使一段受信任代碼獲得更大的權限,甚至比調用它的應用程序還要多,可做到臨時訪問更多的資源。

所以就出現了spring中的典型代碼

 1 if (System.getSecurityManager() != null) {
 2     AccessControlContext acc = getAccessControlContext();
 3     try {
 4         object = AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() {
 5             public Object run() throws Exception {
 6                     return factory.getObject();
 7                 }
 8             }, acc);
 9     }
10     catch (PrivilegedActionException pae) {
11         throw pae.getException();
12     }
13 }else {
14     object = factory.getObject();
15 }

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved