spring bean的生命周期(4)

简介: spring bean生命周期包含bean定义、初始化、使用及销毁等阶段。 bean定义指的是在xml文件中配置bean或bean类中注解bean,bean使用即从springcontext中获取已经初始化的bean并应用的过程,本文主要介绍bean的初始化及销毁阶段。

spring bean生命周期包含bean定义、初始化、使用及销毁等阶段。
bean定义指的是在xml文件中配置bean或bean类中注解bean,bean使用即从springcontext中获取已经初始化的bean并应用的过程,本文主要介绍bean的初始化及销毁阶段。

一、配置实现spring bean初始化、销毁的方式

1、xml文件中配置bean的init-method属性和destroy-method属性

xml配置文件:

<bean name="beanScope" class="com.jsun.test.springDemo.BeanScope" init-method="init_m" destroy-method="destroy_m"></bean>

BeanScope类:

public class BeanScope {

    public BeanScope(){
        System.out.println("BeanScope bean 实例化");
    }

    public void init_m(){
        System.out.println("init-method 初始化");
    }

    public void destroy_m(){
        System.out.println("destroy-method 销毁");
    }
}

从上面执行结果可以看出,bean的实例化构造器要先于初始化方法执行。

同时需要注意的是:如果bean中配置init-method和init-destroy属性,那么在bean类中必须提供对应的初始化和销毁方法,否则将抛出初始化异常:Exception encountered during context initialization

2、配置全局默认的初始化和销毁方法

< beans 
.. .
 default-init-method="init_d_m" default-destroy-method="destroy_d_m">

< bean name="beanScope" class="com.jsun.test.springDemo.BeanScope"></bean>
< /beans>

BeanScope类:

public class BeanScope {

    public BeanScope(){
        System.out.println("BeanScope bean 实例化");
    }

    public void init_d_m(){
        System.out.println("default-init-method 初始化");
    }

    public void destroy_d_m(){
        System.out.println("default-destroy-method 销毁");
    }
}

执行结果与上面情况一致,如果同时配置了bean的初始化、销毁和全局的初始化、销毁方法,那么bean的初始化、销毁方法将覆盖全局配置。【全局配置被覆盖】

另外,配置了全局的初始化、销毁方法,bean类中全局的初始化和销毁方法可以不提供,bean可以被正常初始化。【全局方法可以没有】

3、实现org.springframework.beans.factory.DisposableBean和org.springframework.beans.factory.InitializingBean接口,覆盖实现afterPropertiesSet()和destroy()方法

BeanScope类:

public class BeanScope implements InitializingBean,DisposableBean {

    public BeanScope(){
        System.out.println("BeanScope 实例化");
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        System.out.println("实现接口InitializingBean afterPropertiesSet()初始化");

    }

    @Override
    public void destroy() throws Exception {
        System.out.println("实现接口DisposableBean destroy()销毁");

    }
}

执行结果与上相同

注:如果配置了bean的init-method和init-destroy属性(或同时配置全局的default-init-method和default-destroy-method属性),并实现了上述的初始化和销毁接口,接口的初始化和销毁方法将先于配置的初始化和销毁方法

4、注解配置方式

在指定方法上加上@PostConstruct或@PreDestroy注解来指定该方法作为初始化或销毁方法来使用。

<!-- 开启注解 -->
  <context:annotation-config/>    

  <!-- 配置bean -->
  <bean name="beanScope" class="com.jsun.test.springDemo.BeanScope"></bean>

public class BeanScope{

  public BeanScope(){
      System.out.println("BeanScope bean 实例化");
  }

  //添加@PostConstruct注解,指定当前方法作为初始化方法
  @PostConstruct
  public void init(){
      System.out.println("@PostConstruct注解 初始化");
  }

  //添加@PreDestroy注解,指定当前方法作为销毁方法
  @PreDestroy
  public void destroy(){
      System.out.println("@PreDestroy注解 销毁");
  }
}

二、bean的作用域与生命周期之间的关系以及初始化时机

1、初始化bean实例是时机与bean作用域的关系

(1)、当scope=singleton,即默认情况,会在容器初始化时实例化、初始化。

但如果指定bean节点的lazy-init=”true”,来延迟bean的实例化、初始化,当第一次获取bean时才会初始化bean。

<!-- 使用默认作用域singleton -->
<bean name="beanScope" class="com.jsun.test.springDemo.BeanScope" init-method="init_m" destroy-method="destroy_m" ></bean>

public class BeanScope{

    public BeanScope(){
        System.out.println("BeanScope bean 实例化");
    }

    public void init_m(){
        System.out.println("init-method 初始化");
    }

    public void destroy_m(){
        System.out.println("destroy-method 销毁");
    }


}

    @Test
    public void testScope(){
        System.out.println("获取bean之前");
        BeanScope beanScope = super.getBean("beanScope");
        System.out.println("获取bean的hashCode:"+beanScope.hashCode());
    }

为bean添加lazy-init属性:

<bean name="beanScope" class="com.jsun.test.springDemo.BeanScope" init-method="init_m" destroy-method="destroy_m" lazy-init="true"></bean>

(2)、当scope=prototype时,也会延迟初始化bean,相当于设置lazy-init=”true”,即第一次请求该bean时才初始化(如调用getBean()方法时);并且当spring容器关闭时,destroy()销毁方法也不会被调用。

<bean name="beanScope" class="com.jsun.test.springDemo.BeanScope" init-method="init_m" destroy-method="destroy_m" scope="prototype"></bean>

//spring容器关闭了,但是destroy()销毁方法并没有被执行

原因:对于prototype作用域的bean,有一点非常重要,那就是Spring不能对一个prototype bean的整个生命周期负责:容器在初始化、配置、装饰或者是装配完一个prototype实例后,将它交给客户端,随后就对该prototype实例不闻不问了。不管何种作用域,容器都会调用所有对象的初始化生命周期回调方法。但对prototype而言,任何配置好的析构生命周期回调方法都将不会被调用。

作者:glowd
原文:https://blog.csdn.net/zengqiang1/article/details/54601184
版权声明:本文为博主原创文章,转载请附上博文链接!

相关文章
|
9月前
|
XML Java 测试技术
Spring IOC—基于注解配置和管理Bean 万字详解(通俗易懂)
Spring 第三节 IOC——基于注解配置和管理Bean 万字详解!
640 26
|
11月前
|
XML 安全 Java
|
11月前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
296 12
|
11月前
|
存储 Java 应用服务中间件
【Spring】IoC和DI,控制反转,Bean对象的获取方式
IoC,DI,控制反转容器,Bean的基本常识,类注解@Controller,获取Bean对象的常用三种方式
399 12
|
11月前
|
XML Java 数据格式
Spring容器Bean之XML配置方式
通过对以上内容的掌握,开发人员可以灵活地使用Spring的XML配置方式来管理应用程序的Bean,提高代码的模块化和可维护性。
355 6
|
缓存 Java Spring
实战指南:四种调整 Spring Bean 初始化顺序的方案
本文探讨了如何调整 Spring Boot 中 Bean 的初始化顺序,以满足业务需求。文章通过四种方案进行了详细分析: 1. **方案一 (@Order)**:通过 `@Order` 注解设置 Bean 的初始化顺序,但发现 `@PostConstruct` 会影响顺序。 2. **方案二 (SmartInitializingSingleton)**:在所有单例 Bean 初始化后执行额外的初始化工作,但无法精确控制特定 Bean 的顺序。 3. **方案三 (@DependsOn)**:通过 `@DependsOn` 注解指定 Bean 之间的依赖关系,成功实现顺序控制,但耦合性较高。
748 4
实战指南:四种调整 Spring Bean 初始化顺序的方案
|
11月前
|
XML Java 数据格式
🌱 深入Spring的心脏:Bean配置的艺术与实践 🌟
本文深入探讨了Spring框架中Bean配置的奥秘,从基本概念到XML配置文件的使用,再到静态工厂方式实例化Bean的详细步骤,通过实际代码示例帮助读者更好地理解和应用Spring的Bean配置。希望对你的Spring开发之旅有所助益。
477 4
|
XML Java 数据格式
Spring从入门到入土(bean的一些子标签及注解的使用)
本文详细介绍了Spring框架中Bean的创建和使用,包括使用XML配置文件中的标签和注解来创建和管理Bean,以及如何通过构造器、Setter方法和属性注入来配置Bean。
236 9
Spring从入门到入土(bean的一些子标签及注解的使用)
|
11月前
|
安全 Java 开发者
Spring容器中的bean是线程安全的吗?
Spring容器中的bean默认为单例模式,多线程环境下若操作共享成员变量,易引发线程安全问题。Spring未对单例bean做线程安全处理,需开发者自行解决。通常,Spring bean(如Controller、Service、Dao)无状态变化,故多为线程安全。若涉及线程安全问题,可通过编码或设置bean作用域为prototype解决。
277 1
|
XML Java 数据格式
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
Spring 第二节内容补充 关于Bean配置的更多内容和细节 万字详解!
712 18
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)

热门文章

最新文章

下一篇
oss云网关配置