问题来源:在Listener监听器中无法使用Spring容器的@Resource或者@Autowired 注解的方法注入bean,因为,在web Server容器中,无论是Servlet,Filter,还是Listener都不是Spring容器管理的,因此我们都无法在这些类中直接使用Spring注解的方式来注入我们需要的对象。
在这里,Servlet的整个生命周期都是由Servlet容器来处理的。如果把它硬放到Spring容器中去创建,Servlet对象是可被Spring容器建出来,但Servlet容器可能跟本就不知此Servlet存在,因不在它的容器中。所以,servlet交给web server来管理,不要交给spring管理。
我们在Java web 应用中会使用到监听器来完成一些操作,比如说使用Session监听器来监听Session的变化,通常情况下我们会用javaee规范中的Listener去实现,例如
-
public
class
TestListener
implements
HttpSessionAttributeListener,ServletContextListener
-
{
-
@Override
-
public
void
attributeAdded
(HttpSessionBindingEvent arg0)
-
{
-
// TODO Auto-generated method stub
-
}
-
@Override
-
public
void
attributeRemoved
(HttpSessionBindingEvent arg0)
-
{
-
// TODO Auto-generated method stub
-
}
-
@Override
-
public
void
attributeReplaced
(HttpSessionBindingEvent arg0)
-
{
-
// TODO Auto-generated method stub
-
}
-
@Override
-
public
void
contextDestroyed
(ServletContextEvent arg0)
-
{
-
// TODO Auto-generated method stub
-
}
-
@Override
-
public
void
contextInitialized
(ServletContextEvent arg0)
-
{
-
// TODO Auto-generated method stub
-
}
-
}
现在,我们想在这个监听器里面注入bean,如果使用Spring注解的方式注入bean如下面这样:
-
@Resource
-
private AdvertisementServiceImpl advertisementServiceImpl;
然而以上代码会在项目启动时抛出空指针异常!AdvertisementServiceImpl的实例并没有成功注入。这是为什么呢?要理解这个问题,首先要区分Listener的生命周期和spring管理的bean的生命周期。
(1)Listener的生命周期是由servlet容器(例如tomcat)管理的,项目启动时上例中的TestListener是由servlet容器实例化并调用其contextInitialized方法,而servlet容器并不认得@Resource注解,因此导致AdvertisementServiceImpl实例注入失败。
(2)而spring容器中的bean的生命周期是由spring容器管理的。
那么该如何在spring容器外面获取到spring容器bean实例的引用呢?这就需要用到spring为我们提供的WebApplicationContextUtils工具类,该工具类的作用是获取到spring容器的引用,进而获取到我们需要的bean实例。代码如下
-
@Override
-
public
void
contextInitialized
(ServletContextEvent sce) {
-
WebApplicationContext
applicationContext
= WebApplicationContextUtils.getWebApplicationContext(sce.getServletContext());
-
this.advertisementServiceImpl=(AdvertisementServiceImpl)applicationContext.getBean(
"advertisementServiceImpl");
-
}
然后在spring配置: