面试官:介绍一下Spring Bean的生命周期和作用域

Spring Bean是Spring框架的核心,理解它的生命周期作用域是面试必考知识点,也是实际开发中避坑的关键。本文用大白话+代码示例,带你轻松掌握这两大核心问题!


一、Spring Bean的生命周期:从出生到销毁

Spring Bean的生命周期可以概括为四大阶段实例化 → 属性赋值 → 初始化 → 销毁。但细节远不止这些,我们通过代码一步步拆解:

1. 实例化(Instantiate)

Spring通过反射调用Bean的构造方法 ,创建一个"原始对象"(此时对象还未填充属性)。
相当于:工厂里造了一辆车的骨架,但还没有装发动机和轮胎。

csharp 复制代码
public class Car {
    public Car() {
        System.out.println("1. 调用构造方法:创建Car实例");
    }
}

2. 属性赋值(Populate Properties)

Spring通过依赖注入(DI) 为Bean的属性赋值(如@Autowired@Value)。
相当于:给车装上发动机、轮胎等零件。

kotlin 复制代码
public class Car {
    @Autowired
    private Engine engine;  // 依赖注入引擎

    public void setEngine(Engine engine) {
        System.out.println("2. 属性赋值:注入Engine");
        this.engine = engine;
    }
}

3. 初始化(Initialize)

这一步是Bean的"成人礼",Spring提供了多种扩展点:

  • BeanPostProcessor接口:在初始化前后插入自定义逻辑。
  • @PostConstruct注解:标记初始化完成后执行的方法。
  • InitializingBean接口 :实现afterPropertiesSet()方法。
typescript 复制代码
public class Car implements InitializingBean {
    @PostConstruct
    public void init() {
        System.out.println("3.1 @PostConstruct:初始化方法");
    }

    @Override
    public void afterPropertiesSet() {
        System.out.println("3.2 InitializingBean:属性赋值后执行");
    }
}

4. 销毁(Destroy)

Bean使用完毕后,Spring会调用销毁方法(仅适用于单例Bean)。

  • @PreDestroy注解:标记销毁前执行的方法。
  • DisposableBean接口 :实现destroy()方法。
typescript 复制代码
public class Car implements DisposableBean {
    @PreDestroy
    public void preDestroy() {
        System.out.println("4.1 @PreDestroy:销毁前执行");
    }

    @Override
    public void destroy() {
        System.out.println("4.2 DisposableBean:销毁方法");
    }
}

完整生命周期流程图

less 复制代码
构造方法 → @Autowired注入 → BeanPostProcessor前置处理
 → @PostConstruct → InitializingBean → 
BeanPostProcessor后置处理 → ...(使用中)
→ @PreDestroy → DisposableBean

二、Spring Bean的作用域:单例还是原型?

Spring Bean的作用域决定了Bean的创建方式和生命周期,常用作用域有两种:

1. 单例(Singleton)

  • 特点:整个Spring容器中,Bean只有一个实例(默认作用域)。

  • 适用场景:无状态的工具类、服务类(如Service、DAO)。

  • 配置方式

    less 复制代码
    @Scope("singleton")
    @Service
    public class UserService { ... }

2. 原型(Prototype)

  • 特点:每次请求Bean时,都会创建一个新实例。

  • 适用场景:需要保持状态的Bean(如用户购物车)。

  • 配置方式

    less 复制代码
    @Scope("prototype")
    @Component
    public class ShoppingCart { ... }

其他作用域(Web应用专用)

作用域 描述
request 一次HTTP请求创建一个Bean
session 一次用户会话创建一个Bean
application 整个Web应用共享一个Bean

三、常见问题及避坑指南

问题1:单例Bean中注入原型Bean会失效?

现象 :单例Bean中注入的原型Bean只会初始化一次。
原因 :单例Bean在初始化时已经完成依赖注入,后续请求不会重新创建原型Bean。
解决方案 :使用@Lookup注解或ApplicationContext.getBean()手动获取原型Bean。


问题2:@PostConstruct和InitializingBean的执行顺序?

答案@PostConstruct先于InitializingBean执行。
建议 :优先使用@PostConstruct,代码更简洁。


问题3:Bean的生命周期能定制吗?

能! 通过实现以下接口:

  • BeanPostProcessor:干预所有Bean的初始化过程。
  • BeanFactoryPostProcessor:修改Bean的定义信息。

总结

  • 生命周期 :记住四大阶段(实例化 → 属性注入 → 初始化 → 销毁),善用@PostConstruct@PreDestroy
  • 作用域:单例省资源,原型保状态,Web作用域按需选。
  • 避坑:单例中注入原型Bean需特殊处理,避免状态污染。

相关推荐
weixin_437398211 小时前
RabbitMQ深入学习
java·分布式·后端·spring·spring cloud·微服务·rabbitmq
bing_1582 小时前
Spring MVC 视图解析器 (ViewResolver) 如何配置? Spring Boot 是如何自动配置常见视图解析器的?
spring boot·spring·mvc
残花月伴6 小时前
springCloud/Alibaba常用中间件之Nacos服务注册与发现
spring·spring cloud·中间件
Kx…………6 小时前
Java EE(Spring+Spring MVC+MyBatis)从入门到精通企业级应用开发教程——1初识MyBatis框架
学习·spring·java-ee·mvc·mybatis
超级无敌永恒暴龙战士6 小时前
SpringMVC-执行流程
spring·servlet·springmvc
黄俊懿8 小时前
【深入理解SpringCloud微服务】手写实现一个微服务分布式事务组件
java·分布式·后端·spring·spring cloud·微服务·架构师
有梦想的攻城狮9 小时前
spring中的@Inject注解详情
java·后端·spring·inject
Top`9 小时前
服务预热原理
java·后端·spring
loveLifeLoveCoding10 小时前
springboot 加载 tomcat 源码追踪
java·spring boot·spring
幽络源小助理10 小时前
SpringBoot框架开发网络安全科普系统开发实现
java·spring boot·后端·spring·web安全