Spring执行流程和Bean的生命周期

1、Spring执行流程

Spring执行流程(Bean执行流程):1、在启动类中遇到了ApplicationContext的时候就会启动 Spring 容器 -> 2、根据容器设置的配置文件,去找相应的配置文件; ->3、如果有base-package,那么就去base-package里循环找所有的类,看有没有五大类注解和方法注解; ->4、 如果有注解,进行初始化和属性依赖赋值

在这里插入图片描述

2、Bean的生命周期(重点)

所谓生命周期是指一个对象从创建到销毁的过程。

Bean的生命周期可分为以下5大部分:

  1. 实例化Bean:为Bean分配空间

  2. **设置属性:**为Bean注入属性和装配属性

  3. Bean初始化:

    • 实现各种通知方法:BeanNameAware、BeanFactoryAware、ApplicationContextAware的接口方法。

    • 执行前置方法(BeanPostProcessor接口)

    • 执行初始化方法(设置了才会执行)

      • 注解方式:@PostConstruct,依赖注入后被执行(优先级高,主流方式)
      • xml方式:init-method ⽅法
    • 执行后置方法(BeanPostProcessor接口)

4.使用Bean

5.销毁Bean:@PreDestroy、DisposableBean 接⼝⽅法、destroy-method。

图示化流程:
在这里插入图片描述

伪代码流程(xml方式):

public class BeanLife implements BeanNameAware, BeanFactoryAware, BeanPostProcessor {
    /*
        各种通知
     */
    @Override
    public void setBeanName(String s) {
        //s 是设置的Bean名称
        System.out.println("执行了通知:BeanNameAware - > BeanName = "+s);
    }
    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
        System.out.println("执行了通知:BeanFactoryAware");
    }
    /*
        前置方法 :默认情况不会显式调用
     */
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("执行了前置方法:postProcessBeforeInitialization");
        return bean;
    }
    /*
        初始化
     */
    public void myInit(){
        System.out.println("执行了xml方式的初始化 - > init-method");
    }
    @PostConstruct
    public void doPostConstruct(){
        System.out.println("执行了注解的初始化 - > PostConstruct");
    }
    /*
        后置方法 :默认情况不会显式调用
     */

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("执行了后置方法:postProcessAfterInitialization");
        return bean;
    }

    //使用Bean
    public void sayHi(){
        System.out.println("使用Bean -> sayHi()");
    }
    
    /*
        销毁方法
     */
    @PreDestroy
    public void doPreDestroy(){
        System.out.println("执行了销毁方法:PreDestroy");
    }


}

前置方法和后置方法: 默认不会显式调用

​ 1.可以把作用域改为prototype

xml配置:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:content="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

    <bean id="beanLife" class="com.spring.demo.BeanLife" scope="prototype" init-method="myInit"></bean>
</beans>

启动类:

public class App {
    public static void main(String[] args) {
        //此处使用ClassPathXmlApplicationContext,因为ApplicationContext没有销毁方法
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.config.xml");
        BeanLife beanLife = context.getBean("beanLife",BeanLife.class);
       //使用bean
        beanLife.sayHi();
        //销毁容器(bean也就销毁了)
        context.destroy();
    }
}

执行结果:
在这里插入图片描述

2.1、实例化和初始化的区别

实例化和属性设置是java级别的系统“事件”,其操作过程不可人为干预和修改;而初始化是给开发者提供的,可以在实例化之后,类加载完成之前进行自定义“事件”处理。

2.2、为什么先设置属性再进行初始化呢?

因为在初始化阶段可能会用到bean的属性和方法;如果不先设置属性,先初始化,那么在初始化阶段使用属性,就报错了;因为属性此时还没装配呢。

@Controller
public class BController {
    @Autowired
    private User user1;//设置属性

    @PostConstruct//初始化
    public void say(){
        //再初始化中使用了user1里的属性
        System.out.println(user1.toString());
    }
}
vate User user1;//设置属性

    @PostConstruct//初始化
    public void say(){
        //再初始化中使用了user1里的属性
        System.out.println(user1.toString());
    }
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

bushi橙子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值