作者:江南白衣
Spring再強大,也要面對降臨的問題--因為Spring不是Weblogic、Tomcat般的頂層容器,Servlet和EJB對象不由它創建,所以它必須要降臨到Weblogic、Tomcat所在的位面。
初學者一般不用管那么多,照著Spring+hibernate+Struts之類的Sample就做了,但慢慢的,也許就要開始在jsp+javabean體系,土制框架,singleton類等環境下使用Spring了。
《Professional Java Development with the Spring Framework》第3章有"Managing the Containe"一節講這個問題。一般可以分為直接召喚系與IoC fashion兩類。
1.直接召喚系--Singleton的Application Context
最簡單的,就像在UnitTest里那樣,直接構造Application Context:
ApplicationContext ctx = new ClasspathXmlApplicationContext("ApplicationContext.xml");
在Web環境里,會使用ContextLoader構造ApplicationContext后,壓進Servlet Context。
由ContextLoaderListener或ContextLoaderServlet,在Web應用啟動時完成。
然后在Jsp/Servelet中,可以通過Servlet Context取得ApplicationContext:
ApplicationContext context = WebApplicationContextUtils.getWebApplicationContext(application);
但像singleton類或者EJB中,就沒有Servlet Context可用了。
如果全部像UnitTest那樣直接構造,速度就會很不堪。自然的,就想到把ApplicationContext做成單例。
Spring提供了
ContextSingletonBeanFactoryLocator這樣的物體。
先搞一個beanRefFactory.xml,里面寫上所有的applcationContext-*.xml文件名,并把Context命名為"default-context":
<beans>
<bean id="default-context" class="org.springframework.context.support.ClassPathXmlApplicationContext">
<constructor-arg>
<list> <value>applicationContext.xml</value></list>
</constructor-arg>
</bean>
</beans>
然后讓loactor去找它,但代碼有點長:
BeanFactoryReference bfr = DefaultLocatorFactory.getInstance().useBeanFactory("default-context");
BeanFactory factory = bfr.getFactory();
MyService myService = factory.getBean("myService");
bfr.release();
// now use myService
上面的代碼實在是太靈活,太麻煩了。
還不如自己實現一個簡單的Singleton,擴展ContextLoaderListener類,在Web系統啟動時壓入Singleton。
新的ContextLoaderListener類重載如下,ContextUtil中包含一個靜態的ApplicationContext變量:
public void contextInitialized(ServletContextEvent event)
{
super.contextInitialized(event);
ServletContext context = event.getServletContext();
ApplicationContext ctx = WebApplicationContextUtils.getRequiredWebApplicationContext(context);
ContextUtil.setContext(ctx);
}
用家可直接取用:
ApplicationContext context = ContextUtil.getContext();
2.IoC fashion 如果所有地方都使用直接召喚系,那就反而是在打Rod的耳光了。因為他一直都反對代碼與框架深耦合的。
所以,更好的方法是寫一些glue code、base class來完成Spring的降臨,而不讓應用代碼察覺Spring Application Context的存在。
不過,因為各個框架的結構不同,Rod也沒辦法講出一個通用的整合方法,所以建議大家盡量學習已整合的各種框架,如Spring MVC、Struts的種種方式,寫出自己的簡單整合代碼來。
只有不確定的調用某些Singleton類,不適合過早ioc的情況,可以使用直接召喚系。