我們知道,在Spring2.0中,除了singleton及prototype兩種類型的Bean以外。默認情況下還增加了request、session及global session三種類型的Bean,增加的三種類型的Bean主要應用於Web應用程序中。本文不打算分析三種類型的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工廠。