• bean的生命周期


    目录

            一,bean的初始化过程

    二,bean的单例和多例模式


    一,bean的初始化过程

    如图所示

     1、通过XML、Java annotation(注解)以及Java Configuration(配置类)
    等方式加载Spring Bean

    2、BeanDefinitionReader:解析Bean的定义。在Spring容器启动过程中,
    会将Bean解析成Spring内部的BeanDefinition结构;
    理解为:将spring.xml中的标签转换成BeanDefinition结构
    有点类似于XML解析

     

    3、BeanDefinition:包含了很多属性和方法。例如:id、class(类名)、
    scope、ref(依赖的bean)等等。其实就是将bean(例如)的定义信息
    存储到这个对应BeanDefinition相应的属性中

    4、BeanFactoryPostProcessor:是Spring容器功能的扩展接口。
    注意:
    ①、BeanFactoryPostProcessor在spring容器加载完BeanDefinition之后,
    在bean实例化之前执行的
    ②、对bean元数据(BeanDefinition)进行加工处理,也就是BeanDefinition
    属性填充、修改等操作

    测试:

    1. package com.zking.beanLife;
    2. public class Demo1 {
    3. public static void main(String[] args) {
    4. Person p=new Person();
    5. p.setSex("男");
    6. System.out.println(p.getSex());
    7. }
    8. }
    9. class Person{
    10. private String name;
    11. private int age;
    12. private String sex;
    13. public String getName() {
    14. return name;
    15. }
    16. public void setName(String name) {
    17. this.name = name;
    18. }
    19. public int getAge() {
    20. return age;
    21. }
    22. public void setAge(int age) {
    23. this.age = age;
    24. }
    25. public String getSex() {
    26. return sex;
    27. }
    28. public void setSex(String sex) {
    29. this.sex = sex;
    30. }
    31. public Person() {
    32. this.init();
    33. this.name="zs";
    34. this.age=20;
    35. this.sex="未知";
    36. }
    37. public void init() {
    38. }
    39. @Override
    40. public String toString() {
    41. return "Person [name=" + name + ", age=" + age + ", sex=" + sex + "]";
    42. }
    43. }

    运行结果:

    5、BeanFactory:bean工厂。它按照我们的要求生产我们需要的各种各样的bean
    例如:
    BeanFactory -> List
    BeanDefinition(id/class/scope/init-method)

    foreach(BeanDefinition bean : List){
    //根据class属性反射机制实例化对象
    //反射赋值设置属性
    }

    6、Aware感知接口:在实际开发中,经常需要用到Spring容器本身的功能资源
    例如:BeanNameAware、ApplicationContextAware等等
    BeanDefinition 实现了 BeanNameAware、ApplicationContextAware

    7、BeanPostProcessor:后置处理器。在Bean对象实例化和引入注入完毕后,
    在显示调用初始化方法的前后添加自定义的逻辑。(类似于AOP的绕环通知)
    前提条件:如果检测到Bean对象实现了BeanPostProcessor后置处理器才会执行
    Before和After方法
    BeanPostProcessor
    1)Before
    2)调用初始化Bean(InitializingBean和init-method,Bean的初始化才算完成)
    3)After
    完成了Bean的创建工作

    8、destory:销毁

     总结:

    1.通过三种方式(配置文件,注解,配置类)将bean标签转成BeanDefinition
    2.通过BeanFactoryPostProcessor可以在初始化之前修改属性
    3.BeanFactory进行bean实例化,就是生产javabean
    4.Aware感知接口,能够在拿到Spring上下文内部的资源对象
    5.BeanPostProcessor后置处理器,相当于环绕通知

    二、bean的单例和多例模式

     单例模式(单例是默认):

     多例模式:

     案例
    多例模式

     运行结果:

     

     单例模式图文解说:

    1. public class Demo1 {
    2. public static void main(String[] args) {
    3. Person p1 = Person.newInstance();
    4. Person p2 = Person.newInstance();
    5. Person p3 = Person.newInstance();
    6. Person p4 = Person.newInstance();
    7. System.out.println(p1);
    8. System.out.println(p2);
    9. System.out.println(p3);
    10. System.out.println(p4);
    11. }
    12. }
    13. class Person{
    14. private Person() {
    15. }
    16. private final static Person p = new Person();
    17. public static Person newInstance() {
    18. return p;
    19. }
    20. }

     运行结果:

     

    单例模式的弊端:

    当两个对象使用同一个实现类,会造成变量污染

    这里以给小孩买玩具为例子

    单例模式的代码论证

    test测试:

    1. /*
    2. * spring bean的生命週期
    3. * spring bean的單例多例
    4. */
    5. public class Demo2 {
    6. // 体现单例与多例的区别
    7. @Test
    8. public void test1() {
    9. ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
    10. // ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
    11. ParamAction p1 = (ParamAction) applicationContext.getBean("paramAction");
    12. ParamAction p2 = (ParamAction) applicationContext.getBean("paramAction");
    13. // System.out.println(p1==p2);
    14. p1.execute();
    15. p2.execute();
    16. // 单例时,容器销毁instanceFactory对象也销毁;多例时,容器销毁对象不一定销毁;
    17. applicationContext.close();
    18. }
    19. // 体现单例与多例的初始化的时间点 instanceFactory
    20. @Test
    21. public void test2() {
    22. ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
    23. }
    24. // BeanFactory会初始化bean对象,但会根据不同的实现子类采取不同的初始化方式
    25. // 默认情况下bean的初始化,单例模式立马会执行,但是此时XmlBeanFactory作为子类,单例模式下容器创建,bean依赖没有初始化,只有要获取使用bean对象才进行初始化
    26. @Test
    27. public void test3() {
    28. // ClassPathXmlApplicationContext applicationContext = new
    29. // ClassPathXmlApplicationContext("/spring-context.xml");
    30. Resource resource = new ClassPathResource("/spring-context.xml");
    31. BeanFactory beanFactory = new XmlBeanFactory(resource);
    32. // InstanceFactory i1 = (InstanceFactory) beanFactory.getBean("instanceFactory");
    33. }
    34. }

    InstanceFactory

    1. public class InstanceFactory {
    2. public void init() {
    3. System.out.println("初始化方法");
    4. }
    5. public void destroy() {
    6. System.out.println("销毁方法");
    7. }
    8. public void service() {
    9. System.out.println("业务方法");
    10. }
    11. }

     ParamAction:

    1. /**
    2. * 印证单利和多例的区别
    3. * @author Administrator
    4. *
    5. */
    6. public class ParamAction {
    7. private int age;
    8. private String name;
    9. private List hobby;
    10. private int num = 1;
    11. // private UserBiz userBiz = new UserBizImpl1();
    12. public ParamAction() {
    13. super();
    14. }
    15. public ParamAction(int age, String name, List hobby) {
    16. super();
    17. this.age = age;
    18. this.name = name;
    19. this.hobby = hobby;
    20. }
    21. public void execute() {
    22. // userBiz.upload();
    23. // userBiz = new UserBizImpl2();
    24. System.out.println("this.num=" + this.num++);
    25. System.out.println(this.name);
    26. System.out.println(this.age);
    27. System.out.println(this.hobby);
    28. }
    29. }

    spring-context.xml配置:

     test运行结果:

     进入配置,改成多例模式

     再次运行,结果不相同

     总结:

    单例模式下:javaBean的生命周期:容器生,对象生,容器毁,对象毁
    多例模式下JavaBean的生命周期,使用时对象生,死亡跟着jvm垃圾回收机制走
    bean的初始化时间点,除了跟bean管理模式(单例/多例)有关,还跟beanFactory的子类有关

  • 相关阅读:
    NodeMCU ESP8266 读取按键外部输入信号详解(图文并茂)
    从0开始制作微信小程序
    606. 根据二叉树创建字符串
    LeetCode笔记:Biweekly Contest 85
    如何将Python文件生成bat脚本,点击bat自动运行Python脚本
    数据产品经理日常工作
    数据结构 --- 回溯算法
    PWM控制蜂鸣器
    【FreeRTOS】10 事件标志组
    200个经典C语言程序
  • 原文地址:https://blog.csdn.net/weixin_64313980/article/details/126243485