经过了实际问题的解决和深入实验,总结如下。
以下都是文字性、大纲行描述,具体细节可留言提问,博主会耐心解答。
一、 Spring不能通过注解向Servlet中注入实例的原理
想了解此问题的原理,就要了解tomcat启动后 servlet和spring的加载顺讯。
1、 tomcat启动后先加载web.xml文件。web.xml主要配置了servlet 、filter、listenner三种javaee规范的类,加载顺序跟在web.xml文档
中的位置无关。 顺序为 listenner>filter>servlet 。
2、而spring的初始化类为org.springframework.web.context.ContextLoaderListener,就是一个listenner,它是先于servlet加载的。普通servlet和springmvc的入口servlet
的加载顺序,就要看servle的设置了。
3、在 servlet A类上加@service或@controllert等注解时,spring或springmvc会扫面相关包,自动实例化一个servlet 实例A;这个实例A的引用是spring容器管理的。
4、当然 servlet也会在web.xml配置(要不然怎么拦截url),这是tomcat容器会根据servler配置启动时或者第一次请求该url时实例化该servlet 实例B,
这个实例B的引用是tomcat容器管理的。
5、明白了吧,拦截url的servlet和spring依赖注入的servlet不是同一个实例!!所以就产生了不能依赖注入或者注解不起作用的现象。
二、解决办法
1、第一种,如果在servlet里需要用到一个testservice,不需要在testservice上注解@Autowired,可以简单的用spring硬编码的形式获得。
在你自定义的servlet的 重写方法init中加入如下代码
@Override
public void init() throws ServletException {
WebApplicationContext applicationContext = WebApplicationContextUtils.getWebApplicationContext(getServletContext());
this.fileResourceService=applicationContext.getBean("testservice", testservice.class);
//下面是你的其他代码
}
2、第二种方法
也是在你的servlet init方法里加入spring根据注解注入属性的方法。例如你有testService1和testService2两个属性需要注入,则需要在这两个属性上加注解@Autowired
@Override
public void init() throws ServletException {
SpringBeanAutowiringSupport.processInjectionBasedOnServletContext(this,getServletContext());
}
3、第三种方法
当然第三种方法时实质上和第二种方法一样。当servlet比较多的时候,每次修改init()方法会有重复代码。可以新建一个抽象servlet类,来完成注入。