在Spring中,凡是实现ServletContextAware接口的类,都可以取得ServletContext。实现如下:
1
2
3
4
|
private ServletContext application; public void setServletContext(ServletContext servletContext) { this .application = servletContext; } |
那么Spring是在什么时候把ServletContext放置进去的呢?通过对Spring的学习,终于明白了。
在web项目中,Spring容器的加载是通过XmlWebApplicationContext进行的。
它的父类AbstractRefreshableWebApplicationContext,在postProcessBeanFactory方法中进行了如下操作(postProcessBeanFactory方法被AbstractApplicationContext的refresh方法调用)
1
2
3
4
5
|
beanFactory.addBeanPostProcessor( new ServletContextAwareProcessor( this .servletContext, this .servletConfig)); beanFactory.ignoreDependencyInterface(ServletContextAware. class ); beanFactory.ignoreDependencyInterface(ServletConfigAware. class ); WebApplicationContextUtils.registerWebApplicationScopes(beanFactory, this .servletContext); WebApplicationContextUtils.registerEnvironmentBeans(beanFactory, this .servletContext, this .servletConfig); |
代码的第一句就是添加了一个ServletContextAwareProcessor。
该类的postProcessBeforeInitialization方法如下:
1
2
3
4
5
6
7
8
9
|
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { if ( this .servletContext != null && bean instanceof ServletContextAware) { ((ServletContextAware) bean).setServletContext( this .servletContext); } if ( this .servletConfig != null && bean instanceof ServletConfigAware) { ((ServletConfigAware) bean).setServletConfig( this .servletConfig); } return bean; } |
而所有的BeanPostProcessor都将在AbstractAutowireCapableBeanFactory类的initializeBean方法中,通过调用applyBeanPostProcessorsBeforeInitialization方法完成所有实现BeanPostProcessor接口的postProcessBeforeInitialization的调用。
XmlWebApplicationContext使用的BeanFactory是DefaultListableBeanFactory。
DefaultListableBeanFactory继承了AbstractAutowireCapableBeanFactory,因此可以完成上述操作。
如此完成了只要实现了ServletContextAware接口的,都可以获取ServletContext。