Spring面试题:(四)Spring Bean生命周期

Bean生命周期的阶段

  • 实例化
  • (属性填充)
  • 初始化
  • 完成
  • 销毁
    在这里插入图片描述

IoC容器实例化Bean的流程

Bean定义
Bean工厂处理
反射实例化Bean
初始化
完成存储到单例池
在这里插入图片描述
Bean生命周期
在这里插入图片描述

Bean初始化话过程

  • 属性填充
  • aware接口
  • BeanPostProcessor前置处理
  • InitialzingBean接口初始化方法
  • 自定义init方法
  • BeanPostProcessor后置处理
    在这里插入图片描述

Bean初始化话过程-属性填充过程

  • 注入普通属性:set方法
  • 注入Bean:set方法
  • 双向注入:循环依赖
    在这里插入图片描述

Bean初始化话过程-属性填充过程-如何解决循环依赖

一级缓存:单例池,完整的bean
二级缓存:早期的单例池,不完整的bean,且bean被其它对象引用
三级缓存:单例Bean的工厂,不完整的bean,且bean未被引用,使用时通过工厂getObject方法获取bean
bean初始化后先进入三级缓存,如果被引用,则进入二级缓存,初始化结束后进入一级缓存
在这里插入图片描述
UserService和UserDao互相注入的过程:
先查询-级级缓存,查询不到再查询二级缓存,最后查询三级缓存。
在这里插入图片描述

Bean初始化话过程-aware接口

可以通过aware接口让框架注入spring容器对象。
在这里插入图片描述

Bean初始化话过程-BeanPostProcessor

扩展bean。
BeanPostProcessor接口是Spring框架中的一个扩展点,它允许开发人员在Spring容器完成Bean的实例化、配置和其他初始化后,对Bean进行额外的处理。具体来说,BeanPostProcessor接口提供了两个方法:postProcessBeforeInitialization和postProcessAfterInitialization,分别在Bean的初始化前和初始化后被调用。开发人员可以通过实现BeanPostProcessor接口,重写这两个方法,来对Bean进行自定义的初始化和处理操作。

public class MyBeanPostProcessor implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        // 在Bean初始化前进行自定义处理
        if (bean instanceof MyBean) {
            ((MyBean) bean).setName("new name");
        }
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        // 在Bean初始化后进行自定义处理
        if (bean instanceof MyBean) {
            ((MyBean) bean).setId(100);
        }
        return bean;
    }
}

在上面的示例中,我们定义了一个MyBeanPostProcessor类,实现了BeanPostProcessor接口,并重写了postProcessBeforeInitialization和postProcessAfterInitialization方法。在这两个方法中,我们对MyBean类进行了自定义处理,分别修改了它的name和id属性。
在Spring容器启动时,会自动检测到实现了BeanPostProcessor接口的类,并将它们注册为BeanPostProcessor。当Spring容器完成Bean的实例化、配置和其他初始化后,会依次调用所有注册的BeanPostProcessor的postProcessBeforeInitialization和postProcessAfterInitialization方法,对Bean进行自定义处理。

Bean初始化话过程-InitialzingBean

InitalizingBean接口允许bean在容器设置了bean的所有必要属性后执行执行初始化工作,InitalizingBean接口仅有一个afterPropertiesSet方法,在Spring初始化后,执行完所有属性设置方法(setXxx)后将调afterPropertiesSet

Bean初始化话过程-init-method

通常情况下不鼓励bean直接实现InitializingBean,可以使用Spring提供的init-method的功能来执行一个bean 子定义的初始化方法。
先后顺序

  • @PostConstruct注释的方法
  • InitializingBean回调接口定义的afterPropertiesSet()
  • 自定义配置的init()方法

Bean销毁过程

首先判断Bean是否实现了DestructionAwareBeanPostProcessor接口,如果实现了,则会执行DestructionAwareBeanPostProcessor后置处理器的销毁回调方法
其次会判断Bean是否实现了DisposableBean接口,如果实现了将会调用其实现的destroy()方法
最后判断这个Bean是否配置了dlestroy-method等自定义的销毁方法,如果有的话,则会自动调用其配置的销毁方法;

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
面试高级开发的期间整理的面试题目,记录我面试遇到过的spring题目以及答案 目录 spring ThreadLocal的底层对象; 为什么@Service和@Repository放到实现类上面而不是接口类上面; spring 三种注入(就是从spring容器中将bean放入对象属性值中) Spring下描述依赖关系@Resource, @Autowired和@Inject的区别与联系 SpringBeanFactory和ApplicationContext的区别 谈谈Spring IOC的理解,原理与实现? bean的生命周期,详细看上面 SpringBoot自动装配的过程的原理: spring的缓存; spring是如何解决的循环依赖; BeanFactory和FactoryBean有什么区别; Spring中用到的设计模式; SPI 机制(Java SPI 实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制), 很多地方有用到: AOP Spring的AOP的底层实现原理; 为什么jdk动态代理是必须是接口 两种动态代理的区别 AOP实现方式:aop注解或者xml配置;后来工具jar包aspects; aop的属性 事务 事务编码方式: 事务注意事项; 为什么同一个类A调用b方法事务,A方法一定要有事务(编码式的不用) @transaction多个数据源事务怎么指定数据源 传播特性有几种?7种; 某一个事务嵌套另一个事务的时候怎么办? REQUIRED_NEW和REQUIRED区别 Spring的事务是如何回滚的,实现原理; 抽象类和接口的区别,什么时候用抽象类什么时候用接口; StringBuilder和StringBuffer的区别 java值传递和引用传递

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值