1. 描述
spring中提供了两种主要方式实例化bean,一种为配置文件方式,另一种为注解的形式。但如果配置文件配置不合理或者注解使用不恰当,就会造成一个bean会被多次初始化的现象发生。此时会造成一种资源的浪费,严重时甚至会影响系统的性能。但此种问题有很隐蔽,如果不仔细检查,很难发现。本人是因为系统中一个定时任务被重复执行两次,经google,baidu之后才发现此问题。 以下是几种会产生此问题的配置形式。
2. 配置文件问题导致的重复初始化
2.1 问题原因
有时候,我们会将spring的配置和spring mvc的配置放在一个xml文件中,比如叫:applicationContext.xml,此时,我们在web.xml文件中初始化spring容器时,一般会做如下形式的配置
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
<!-- 配置Spring的监听,否则业务层的bean实例无法创建,也就不能使用spring的ioc了 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<servlet>
<servlet-name>dispatcherServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:/applicationContext.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>dispatcherServlet</servlet-name>
<url-pattern>*.do</url-pattern>
</servlet-mapping>
在此处配置中,context-param
用于初始化spring的applicationContext,Servlet
用于初始化spring的DispatcherServlet,此时就会造成applicationContext.xml
中的bean会被重复初始化两次。
1.2 解决方案
此种情况,可以删除context-param
和listener
来达到目的,此时只剩下一个Servlet
的配置,所有bean都会被初始化一次。但是spring中有个父子容器的概念,即listener
中初始化的applicationContext是Servlet
初始化的WebApplicationContext的父容器,**子容器可以访问使用父容器中实例化的bean,但是父容器不能使用子容器中实例化的bean,即不能使用ref子容器中实例化的bean。**所以,此种情况下,最好的解决方法是:将applicationContext.xml
文件拆成两个文件,并根据业务需求划分两个文件中的功能和配置。
3. 注解问题导致的重复初始化
3.1 问题原因
如果在applicationContext.xml
中配置bean,同时用了注解,配置类似如下:
<context:component-scanbase-package="com.zhlong.test"></context:component-scan>
<beanclass="com.zhlong.test.People">
</bean>
同时People类上使用了注解,类似如下:
@Component
public class People {
public People(){
System.out.println("People类被初始化...");
}
}
此时系统启动时,people类会被初始化两次,生成两个people类实例。会发现People类被初始化...
这句话被打印两次。
3.2 解决方案
配置和注解只使用一种方式。根据业务需要配置。<div class="iteye-blog-content-contain" style="font-size: 14px"></div>
https://my.oschina.net/u/575836/blog/552394