想了解此问题的原理,就要了解tomcat启动后 servlet和spring的加载顺序。在TomcatWeb 容器加载一个Web项目的时候:
此处摘自:https://blog.csdn.net/qq_21125183/article/details/86242910
tomcat启动后先加载web.xml文件。web.xml主要配置了servlet 、filter、listenner三种javaEE规范的类,加载顺序跟在web.xml文档中的位置无关。顺序为 listenner>filter>servlet 。
而spring的初始化类为org.springframework.web.context.ContextLoaderListener,就是一个listenner,它是先于servlet加载的。普通servlet和springmvc的入口servlet的加载顺序,就要看servlet的设置了。它们按照在Web.xml中定义的Servlet顺序加载。其中springmvc需要指定org.springframework.web.servlet.DispatcherServlet拦截所有的Web请求。
在 servletA类上加@WebServlet等注解时,spring或springmvc会扫面相关包,自动实例化一个servlet实例A;这个实例A的引用是spring IOC容器管理的。这个时候Spring ContextLoaderListener监听器首先初始化,扫描所有的java包,创建Bean对象。然后Tomcat容器在加载Servlet类,包括我们定义的Servlet以及Spring的DispatcherServlet。
Tomcat容器接下来会在web.xml配置加载Servlet类,这个时候加载DispatcherServle以及我们定义的Servlet类。这是tomcat容器会根据servler配置启动时或者第一次请求该url时实例化我们定义的Web servlet实例B.这个实例B的引用是tomcat容器管理的。
所以最终结果就是:拦截url的servlet和spring依赖注入的servlet不是同一个实例!!所以就产生了不能依赖注入或者注解不起作用的现象。
解决方法
Servlet中init()方法,重新注入Bean
public void init() throws ServletException {
try {
WebApplicationContext webApp= WebApplicationContextUtils.getWebApplicationContext(getServletContext());
resRecvDaoImp= (RepositoryService) webApp.getBean("resRecvDaoImp");
} catch (Exception e) {
e.printStackTrace();
}
}
init-param里配置实现类,导致SqlSessionTemplate总是报空指针无法注入
解决方式:
public SqlSessionTemplate getSqlTempate(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext(
new String[] { "classpath:applicationContext-tag.xml", "classpath:applicationContext-tx.xml" });
SqlSessionTemplate sqlSessionTemplate = (SqlSessionTemplate)applicationContext.getBean("sqlSessionTemplate");
return sqlSessionTemplate;
}