# 《Spring注解驱动开发》第12章:使用@Bean注解指定初始化和销毁的方法

# 写在前面

在【String注解驱动开发专题 (opens new window)】中,前面的文章我们主要讲了有关于如何向Spring容器中注册bean的知识,大家可以到【String注解驱动开发专题 (opens new window)】中系统学习。接下来,我们继续肝Spring,只不过从本篇文章开始,我们就进入Spring容器中有关Bean的生命周期的学习。

项目工程源码已经提交到GitHub:https://github.com/binghe001/spring-annotation (opens new window)

# Bean的生命周期

通常意义上讲的bean的名称周期,指的是bean从创建到初始化,经过一系列的流程,最终销毁的过程。只不过,在Spring中,bean的生命周期是由Spring容器来管理的。在Spring中,我们可以自己来指定bean的初始化和销毁的方法。当我们指定了bean的初始化和销毁方法时,当容器在bean进行到当前生命周期的阶段时,会自动调用我们自定义的初始化和销毁方法。

# 如何定义初始化和销毁方法?

我们已经知道了由Spring管理bean的生命周期时,我们可以指定bean的初始化和销毁方法,那具体该如何定义这些初始化和销毁方法呢?接下来,我们就介绍第一种定义初始化和销毁方法的方式: 通过@Bean注解指定初始化和销毁方法。

如果是使用XML文件的方式配置bean的话,可以在标签中指定bean的初始化和销毁方法,如下所示。

<bean id = "person" class="io.mykit.spring.plugins.register.bean.Person" init-method="init" destroy-method="destroy">
    <property name="name" value="binghe"></property>
    <property name="age" value="18"></property>
</bean>
1
2
3
4

这里,需要注意的是,在我们写的Person类中,需要存在init()方法和destroy()方法。而且Spring中规定,这里的init()方法和destroy()方法必须是无参方法,但可以抛异常。

如果我们使用注解的方式,该如何实现指定bean的初始化和销毁方法呢?接下来,我们就一起来搞定它!!

首先,创建一个名称为Student的类,这个类的实现比较简单,如下所示。

package io.mykit.spring.plugins.register.bean;
/**
 * @author binghe
 * @version 1.0.0
 * @description 测试bean的初始化和销毁方法
 */
public class Student {
    
    public Student(){
        System.out.println("Student类的构造方法");
    }

    public void init(){
        System.out.println("初始化Student对象");
    }

    public void destroy(){
        System.out.println("销毁Student对象");
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

接下来,我们将Student类对象通过注解的方式注册到Spring容器中,具体的做法就是新建一个LifeCircleConfig类作为Spring的配置类,将Student类对象通过LifeCircleConfig类注册到Spring容器中,LifeCircleConfig类的代码如下所示。

package io.mykit.spring.plugins.register.config;

import io.mykit.spring.plugins.register.bean.Student;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * @author binghe
 * @version 1.0.0
 * @description Bean的生命周期
 */
@Configuration
public class LifeCircleConfig {
    @Bean
    public Student student(){
        return new Student();
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

接下来,我们就新建一个BeanLifeCircleTest类来测试容器中的Student对象,BeanLifeCircleTest类的部分代码如下所示。

package io.mykit.spring.test;

import io.mykit.spring.plugins.register.config.LifeCircleConfig;
import org.junit.Test;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

/**
 * @author binghe
 * @version 1.0.0
 * @description 测试bean的生命周期
 */
public class BeanLifeCircleTest {

    @Test
    public void testBeanLifeCircle01(){
        //创建IOC容器
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(LifeCircleConfig.class);
        System.out.println("容器创建完成...");
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

在前面的文章中,我们说过:对于单实例bean对象来说,在Spring容器创建完成后,就会对单实例bean进行实例化。那么,我们先来运行下BeanLifeCircleTest类中的testBeanLifeCircle01()方法,输出的结果信息如下所示。

Student类的构造方法
容器创建完成...
1
2

可以看到,在Spring容器创建完成时,自动调用单实例bean的构造方法,对单实例bean进行了实例化操作。

总之:对于单实例bean来说,在Spring容器启动的时候创建对象;对于多实例bean来说,在每次获取bean的时候创建对象。

现在,我们在Student类中指定了init()方法和destroy()方法,那么,如何让Spring容器知道Student类中的init()方法是用来执行对象的初始化操作,而destroy()方法是用来执行对象的销毁操作呢?如果是使用XML文件配置的话,我们可以使用如下配置来实现。

<bean id="student" class="io.mykit.spring.plugins.register.bean.Student" init-method="init" destroy-method="destroy"></bean>
1

如果我们在@Bean注解中该如何实现呢?其实就更简单了,我们来看下@Bean注解的源码,如下所示。

package org.springframework.context.annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import org.springframework.beans.factory.annotation.Autowire;
import org.springframework.beans.factory.support.AbstractBeanDefinition;
import org.springframework.core.annotation.AliasFor;

@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Bean {

	@AliasFor("name")
	String[] value() default {};

	@AliasFor("value")
	String[] name() default {};

	@Deprecated
	Autowire autowire() default Autowire.NO;

	boolean autowireCandidate() default true;

	String initMethod() default "";

	String destroyMethod() default AbstractBeanDefinition.INFER_METHOD;

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33

看到@Bean注解的源码,相信小伙伴们会有种豁然开朗的感觉:没错,就是使用@Bean注解的initMethod属性和destroyMethod属性来指定bean的初始化方法和销毁方法。

所以,我们在LifeCircleConfig类中的@Bean注解中指定initMethod属性和destroyMethod属性,如下所示。

@Bean(initMethod = "init", destroyMethod = "destroy")
public Student student(){
    return new Student();
}
1
2
3
4

此时,我们再来运行BeanLifeCircleTest类中的testBeanLifeCircle01()方法,输出的结果信息如下所示。

Student类的构造方法
初始化Student对象
容器创建完成...
1
2
3

从输出结果可以看出,在Spring容器中,先是调用了Student类的构造方法来创建Student对象,接下来调用了Student对象的init()方法来进行初始化。

那小伙伴们可能会问,运行上面的代码没有打印出bean的销毁方法中的信息啊,那什么时候执行bean的销毁方法呢? 这个问题问的很好, bean的销毁方法是在容器关闭的时候调用的。

接下来,我们在BeanLifeCircleTest类中的testBeanLifeCircle01()方法中,添加关闭容器的代码,如下所示。

@Test
public void testBeanLifeCircle01(){
    //创建IOC容器
    AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(LifeCircleConfig.class);
    System.out.println("容器创建完成...");
    context.close();
}
1
2
3
4
5
6
7

我们再来运行BeanLifeCircleTest类中的testBeanLifeCircle01()方法,输出的结果信息如下所示。

Student类的构造方法
初始化Student对象
容器创建完成...
销毁Student对象
1
2
3
4

可以看到,此时输出了对象的销毁方法中的信息,说明执行了对象的销毁方法。

# 指定初始化和销毁方法的使用场景

一个典型的使用场景就是对于数据源的管理。例如,在配置数据源时,在初始化的时候,对很多的数据源的属性进行赋值操作;在销毁的时候,我们需要对数据源的连接等信息进行关闭和清理。此时,我们就可以在自定义的初始化和销毁方法中来做这些事情!

# 初始化和销毁方法调用的时机

  • bean对象的初始化方法调用的时机:对象创建完成,如果对象中存在一些属性,并且这些属性也都赋值好之后,会调用bean的初始化方法。对于单实例bean来说,在Spring容器创建完成后,Spring容器会自动调用bean的初始化和销毁方法;对于单实例bean来说,在每次获取bean对象的时候,调用bean的初始化和销毁方法。
  • bean对象的销毁方法调用的时机:对于单实例bean来说,在容器关闭的时候,会调用bean的销毁方法;对于多实例bean来说,Spring容器不会管理这个bean,也不会自动调用这个bean的销毁方法。不过,小伙伴们可以手动调用多实例bean的销毁方法。

前面,我们已经说了单实例bean的初始化和销毁方法。接下来,我们来说下多实例bean的初始化和销毁方法。我们将Student对象变成多实例bean来验证下。接下来,我们在LifeCircleConfig类的student()方法上通过@Scope注解将Student对象设置成多实例bean,如下所示。

@Scope("prototype")
@Bean(initMethod = "init", destroyMethod = "destroy")
public Student student(){
    return new Student();
}
1
2
3
4
5

接下来,我们再来运行BeanLifeCircleTest类中的testBeanLifeCircle01()方法,输出的结果信息如下所示。

容器创建完成...
1

可以看到,当我们将Student对象设置成多实例bean,并且没有获取bean实例对象时,Spring容器并没有执行bean的构造方法、初始化方法和销毁方法。

说到这,我们就在BeanLifeCircleTest类中的testBeanLifeCircle01()方法中添加一行获取Student对象的代码,如下所示。

@Test
public void testBeanLifeCircle01(){
    //创建IOC容器
    AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(LifeCircleConfig.class);
    System.out.println("容器创建完成...");
    context.getBean(Student.class);
    context.close();
}
1
2
3
4
5
6
7
8

此时,我们再来运行BeanLifeCircleTest类中的testBeanLifeCircle01()方法,输出的结果信息如下所示。

容器创建完成...
Student类的构造方法
初始化Student对象
1
2
3

可以看到,此时,结果信息中输出了构造方法和初始化方法中的信息。但是当容器关闭时,并没有输出bean的销毁方法中的信息。

这是因为 将bean设置成多实例时,Spring不会自动调用bean对象的销毁方法。至于多实例bean对象何时销毁,那就是程序员自己的事情了!!Spring容器不再管理多实例bean。

好了,咱们今天就聊到这儿吧!别忘了给个在看和转发,让更多的人看到,一起学习一起进步!!

项目工程源码已经提交到GitHub:https://github.com/binghe001/spring-annotation (opens new window)

# 星球服务

加入星球,你将获得:

1.项目学习:微服务入门必备的SpringCloud Alibaba实战项目、手写RPC项目—所有大厂都需要的项目【含上百个经典面试题】、深度解析Spring6核心技术—只要学习Java就必须深度掌握的框架【含数十个经典思考题】、Seckill秒杀系统项目—进大厂必备高并发、高性能和高可用技能。

2.框架源码:手写RPC项目—所有大厂都需要的项目【含上百个经典面试题】、深度解析Spring6核心技术—只要学习Java就必须深度掌握的框架【含数十个经典思考题】。

3.硬核技术:深入理解高并发系列(全册)、深入理解JVM系列(全册)、深入浅出Java设计模式(全册)、MySQL核心知识(全册)。

4.技术小册:深入理解高并发编程(第1版)、深入理解高并发编程(第2版)、从零开始手写RPC框架、SpringCloud Alibaba实战、冰河的渗透实战笔记、MySQL核心知识手册、Spring IOC核心技术、Nginx核心技术、面经手册等。

5.技术与就业指导:提供相关就业辅导和未来发展指引,冰河从初级程序员不断沉淀,成长,突破,一路成长为互联网资深技术专家,相信我的经历和经验对你有所帮助。

冰河的知识星球是一个简单、干净、纯粹交流技术的星球,不吹水,目前加入享5折优惠,价值远超门票。加入星球的用户,记得添加冰河微信:hacker_binghe,冰河拉你进星球专属VIP交流群。

# 星球重磅福利

跟冰河一起从根本上提升自己的技术能力,架构思维和设计思路,以及突破自身职场瓶颈,冰河特推出重大优惠活动,扫码领券进行星球,直接立减149元,相当于5折, 这已经是星球最大优惠力度!


领券加入星球,跟冰河一起学习《SpringCloud Alibaba实战》、《手撸RPC专栏》和《Spring6核心技术》,更有已经上新的《大规模分布式Seckill秒杀系统》,从零开始介绍原理、设计架构、手撸代码。后续更有硬核中间件项目和业务项目,而这些都是你升职加薪必备的基础技能。

100多元就能学这么多硬核技术、中间件项目和大厂秒杀系统,如果是我,我会买他个终身会员!

# 其他方式加入星球

特别提醒: 苹果用户进圈或续费,请加微信 hacker_binghe 扫二维码,或者去公众号 冰河技术 回复 星球 扫二维码加入星球。

# 星球规划

后续冰河还会在星球更新大规模中间件项目和深度剖析核心技术的专栏,目前已经规划的专栏如下所示。

# 中间件项目

  • 《大规模分布式定时调度中间件项目实战(非Demo)》:全程手撸代码。
  • 《大规模分布式IM(即时通讯)项目实战(非Demo)》:全程手撸代码。
  • 《大规模分布式网关项目实战(非Demo)》:全程手撸代码。
  • 《手写Redis》:全程手撸代码。
  • 《手写JVM》全程手撸代码。

# 超硬核项目

  • 《从零落地秒杀系统项目》:全程手撸代码,在阿里云实现压测(已上新)。
  • 《大规模电商系统商品详情页项目》:全程手撸代码,在阿里云实现压测。
  • 其他待规划的实战项目,小伙伴们也可以提一些自己想学的,想一起手撸的实战项目。。。

既然星球规划了这么多内容,那么肯定就会有小伙伴们提出疑问:这么多内容,能更新完吗?我的回答就是:一个个攻破呗,咱这星球干就干真实中间件项目,剖析硬核技术和项目,不做Demo。初衷就是能够让小伙伴们学到真正的核心技术,不再只是简单的做CRUD开发。所以,每个专栏都会是硬核内容,像《SpringCloud Alibaba实战》、《手撸RPC专栏》和《Spring6核心技术》就是很好的示例。后续的专栏只会比这些更加硬核,杜绝Demo开发。

小伙伴们跟着冰河认真学习,多动手,多思考,多分析,多总结,有问题及时在星球提问,相信在技术层面,都会有所提高。将学到的知识和技术及时运用到实际的工作当中,学以致用。星球中不少小伙伴都成为了公司的核心技术骨干,实现了升职加薪的目标。

# 联系冰河

# 加群交流

本群的宗旨是给大家提供一个良好的技术学习交流平台,所以杜绝一切广告!由于微信群人满 100 之后无法加入,请扫描下方二维码先添加作者 “冰河” 微信(hacker_binghe),备注:星球编号

冰河微信

# 公众号

分享各种编程语言、开发技术、分布式与微服务架构、分布式数据库、分布式事务、云原生、大数据与云计算技术和渗透技术。另外,还会分享各种面试题和面试技巧。内容在 冰河技术 微信公众号首发,强烈建议大家关注。

公众号:冰河技术

# 视频号

定期分享各种编程语言、开发技术、分布式与微服务架构、分布式数据库、分布式事务、云原生、大数据与云计算技术和渗透技术。另外,还会分享各种面试题和面试技巧。

视频号:冰河技术

# 星球

加入星球 冰河技术 (opens new window),可以获得本站点所有学习内容的指导与帮助。如果你遇到不能独立解决的问题,也可以添加冰河的微信:hacker_binghe, 我们一起沟通交流。另外,在星球中不只能学到实用的硬核技术,还能学习实战项目

关注 冰河技术 (opens new window)公众号,回复 星球 可以获取入场优惠券。

知识星球:冰河技术