spring中bean被多次实例化问题

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-paramlistener来达到目的,此时只剩下一个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

猜你喜欢

转载自m635674608.iteye.com/blog/2398975