Spring2.0中新的Bean類型實現原理
我們知道,在Spring2.0中,除了singleton及prototype兩種類型的Bean以外。默認情況下還增加了request、session及global session三種類型的Bean,增加的三種類型的Bean主要應用于Web應用程序中。本文不打算分析三種類型的Bean的用法,只是簡單分析框架的實現原理。
Spring2.0中新增了一個用來表示Bean范圍的Scope接口
在容器ConfigurableBeanFactory接口中定義了Bean工廠有關Scope注冊的相關方法,使得可往Bean工廠中加入新類型的Bean。
在AbstractFactoryBean的getBean方法中實現了對特定Scope Bean支持,核心代碼摘要:
默認情況下,低層的Bean工廠中只支持singleton及prototype兩種類型的Bean。當把scope設置成request及session時將會出現不能正確識別Scope的錯誤。這是因為普通的Bean工廠都沒有注冊新的Scope。只有在WebApplicationContext中注冊才注冊了新類型的Bean。
下面看實現注冊Scope的代碼:
在WebApplicationContext中定義常量
然后在所有類型的Web應用上下文的實現中,都在Bean工廠的攔載過程中通過postProcessBeanFactory方法來注冊新類型Scope,如GenericWebApplicationContext、StaticWebApplicationContext、AbstractRefreshableWebApplicationContext等WebApplication應用上下文實現中。
結合上面的代碼,現在應該明白為什么只有在Web應用上下文中才能使用新增加三種類型的Bean了吧。當然,由于有了Scope,我們也可以非常輕松的實現我們自己的Scope,增加新用戶自定義類型的Bean,然后設計出一個適合我們自己的Bean工廠。
Spring2.0中新增了一個用來表示Bean范圍的Scope接口
| public interface Scope { Object get(String name, ObjectFactory objectFactory);//根據名稱及創建工廠得到一個Bean實例 Object remove(String name);//刪除一個指定名稱的Bean } |
在容器ConfigurableBeanFactory接口中定義了Bean工廠有關Scope注冊的相關方法,使得可往Bean工廠中加入新類型的Bean。
| public interface ConfigurableBeanFactory extends HierarchicalBeanFactory, void registerScope(String scopeName, Scope scope);//往Bean工廠中添加一個新的范圍(默認只有兩種范圍: singleton及prototype) void destroyScopedBean(String beanName);//銷毀B ean工廠中范圍Bean } |
在AbstractFactoryBean的getBean方法中實現了對特定Scope Bean支持,核心代碼摘要:
| String scopeName = mergedBeanDefinition.getScope(); //取得當前Bean的范圍,也即在定義中的scope=”request”的部分。 Scope scope = (Scope) this.scopes.get(scopeName);//得到Bean工廠中的范圍處理器 if (scope == null) { throw new IllegalStateException("No Scope registered for scope '" + scopeName + "'"); } try { //使用scope.get(beanName,ObjectFactory)從指定的范圍中得到或創建Bean實例 Object scopedInstance = scope.get(beanName, new ObjectFactory() { public Object getObject() throws BeansException { beforePrototypeCreation(beanName);//前攔截 try { return createBean(beanName, mergedBeanDefinition, args); //調用子類的createBean實現真正的Bean創建工作 } finally { afterPrototypeCreation(beanName);//后攔截 } } }); bean = getObjectForBeanInstance(scopedInstance, name, mergedBeanDefinition);//返回正確類型的Bean實例 } catch (IllegalStateException ex) { throw new BeanCreationException(beanName, "Scope '" + scopeName + "' is not active", ex); } |
默認情況下,低層的Bean工廠中只支持singleton及prototype兩種類型的Bean。當把scope設置成request及session時將會出現不能正確識別Scope的錯誤。這是因為普通的Bean工廠都沒有注冊新的Scope。只有在WebApplicationContext中注冊才注冊了新類型的Bean。
下面看實現注冊Scope的代碼:
在WebApplicationContext中定義常量
| public interface WebApplicationContext extends ApplicationContext { String SCOPE_REQUEST = "request"; String SCOPE_SESSION = "session"; String SCOPE_GLOBAL_SESSION = "globalSession"; } |
然后在所有類型的Web應用上下文的實現中,都在Bean工廠的攔載過程中通過postProcessBeanFactory方法來注冊新類型Scope,如GenericWebApplicationContext、StaticWebApplicationContext、AbstractRefreshableWebApplicationContext等WebApplication應用上下文實現中。
| protected void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) { beanFactory.registerScope(SCOPE_REQUEST, new RequestScope());//注冊request類型的Bean beanFactory.registerScope(SCOPE_SESSION, new SessionScope(false));//注冊session類型的Bean beanFactory.registerScope(SCOPE_GLOBAL_SESSION, new SessionScope(true));//注冊glogalsession的Bean } |
結合上面的代碼,現在應該明白為什么只有在Web應用上下文中才能使用新增加三種類型的Bean了吧。當然,由于有了Scope,我們也可以非常輕松的實現我們自己的Scope,增加新用戶自定義類型的Bean,然后設計出一個適合我們自己的Bean工廠。