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