• bean的生命周期


    目录

    一、bean的生命周期原理

    1、Spring Bean的生命周期

    2、bean生命周期原理图

     3、bean简单原理

    二、单例多例理论

    1、介绍单例多例

    2、单例多例区别

    三、代码论证

    1、初始化javabean

    2、区别单例多例

    3、 测试类

    4、配置spring

    5、运行

    6、 结论:

    一、bean的生命周期原理

    1、Spring 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相应的属性中

    例如:

    -----> BeanDefinition(id/class/scope)

    4)BeanFactoryPostProcessor:是Spring容器功能的扩展接口。

    注意:

    ①BeanFactoryPostProcessor在spring容器加载完BeanDefinition之后,

    在bean实例化之前执行的

    ②对bean元数据(BeanDefinition)进行加工处理,也就是BeanDefinition

    属性填充、修改等操作

    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

    ①Before

    ②调用初始化Bean(InitializingBean和init-method,Bean的初始化才算完成)

    ③After

    完成了Bean的创建工作

    8)destory:销毁

    2、bean生命周期原理图

     3、bean简单原理

    1.通过三种方式(配置文件、注解、配置类)将bean标签转成beandifinition对象

    2.通过BeanFactoryPostProcessor可以在初始化之前修改属性值

    3.BeanFactory进行bean实例化,其实就是生产javabean

    4.Aware感知接口,能够在拿到Spring上下文中内部的资源对象

    5.BeanPostProcessor后置处理器,相对于环绕通知

    二、单例多例理论

    1、介绍单例多例

    Spring的javaBean是分为两种的,一种是单例,一种是多例;

     我们在Spring配置文件中也是可以配置的: 

    scope="singleton"   ---> 代表单例

    scope="prototype"  ---> 代表多例

    不配置的话默认单例

    1. <bean class="com.zking.biz.impl.UserBizImlp1" id="userBiz">bean>
    2. <bean class="com.zking.biz.impl.UserBizImlp1" id="userBiz" scope="singleton">bean>
    3. <bean class="com.zking.biz.impl.UserBizImlp1" id="userBiz" scope="prototype">bean>

    2、单例多例区别

    普通代码

    1. package com.zking.beanLife;
    2. public class Demo1 {
    3. public static void main(String[] args) {
    4. Person p1 = new Person();
    5. Person p2 = new Person();
    6. Person p3 = new Person();
    7. Person p4 = new Person();
    8. System.out.println(p1);
    9. System.out.println(p2);
    10. System.out.println(p3);
    11. System.out.println(p4);
    12. }
    13. }
    14. class Person{
    15. }

    运行

      一个类如果被使用了一百次会创建多少个对象?如果这个项目一共有一万个类呢? 

    1百万个

    这样极大的占用了我们的内存空间

    用单例模式

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

    运行

     一个类在100个地方使用,使用的但是同一个对象,这个项目一共有1w个类,请问Spring上下文

    要创建多少个对象?

    1w个

    结论:

    单例模式的优点:就是极大的节省了空间,省略了很多的没必要的对象创建。

    所以Spring默认使用单例模式;

    问题·:为什么spring框架作者要提供原型/多例和javabean管理模式进行选择?

    原因:单例模式也有弊端

    弊端:变量会被污染

    三、代码论证

    1、初始化javabean

    1. package com.zking.beanLife;
    2. /**
    3. * 为了印证BeanPostProcessor 初始化javabean
    4. * @author yzq
    5. *
    6. * 2022年8月9日上午10:12:25
    7. */
    8. public class InstanceFactory {
    9. public void init() {
    10. System.out.println("初始化方法");
    11. }
    12. public void destroy() {
    13. System.out.println("销毁方法");
    14. }
    15. public void service() {
    16. System.out.println("业务方法");
    17. }
    18. }

    2、区别单例多例

    1. package com.zking.beanLife;
    2. import java.util.List;
    3. /**
    4. * 印证单例多例模式的区别
    5. * @author yzq
    6. *
    7. * 2022年8月9日上午10:13:56
    8. */
    9. public class ParamAction {
    10. private int age;
    11. private String name;
    12. private List hobby;
    13. private int num = 1;
    14. // private UserBiz userBiz = new UserBizImpl1();
    15. public ParamAction() {
    16. super();
    17. }
    18. public ParamAction(int age, String name, List hobby) {
    19. super();
    20. this.age = age;
    21. this.name = name;
    22. this.hobby = hobby;
    23. }
    24. public void execute() {
    25. // userBiz.upload();
    26. // userBiz = new UserBizImpl2();
    27. System.out.println("this.num=" + this.num++);
    28. System.out.println(this.name);
    29. System.out.println(this.age);
    30. System.out.println(this.hobby);
    31. }
    32. }

    3、 测试类

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

    4、配置spring

    1. <beans xmlns="http://www.springframework.org/schema/beans"
    2. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    3. xmlns:aop="http://www.springframework.org/schema/aop"
    4. xmlns:context="http://www.springframework.org/schema/context"
    5. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
    6. http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
    7. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
    8. <bean id="paramAction" class="com.zking.beanLife.ParamAction">
    9. <constructor-arg name="name" value="三丰">constructor-arg>
    10. <constructor-arg name="age" value="21">constructor-arg>
    11. <constructor-arg name="hobby">
    12. <list>
    13. <value>抽烟value>
    14. <value>烫头value>
    15. <value>大保健value>
    16. list>
    17. constructor-arg>
    18. bean>
    19. <bean id="instanceFactory" class="com.zking.beanLife.InstanceFactory"
    20. scope="prototype" init-method="init" destroy-method="destroy">bean>
    21. beans>

    5、运行

     由此我们可以发现他们的num值已经不一样了就当于被污染了

    我们改成多例模式在了运行一次

     运行

    6、 结论:

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

  • 相关阅读:
    FreeRTOS 计数型信号量 详解
    Windows自动挂载smb/nfs到本地
    为视觉语言多模态模型进行偏好优化
    TK爆品剖析 水晶首饰降临节日历持续火爆TikTok,独立站卖到断货
    faust勒索病毒:最新变种[nicetomeetyou@onionmail.org].faust袭击了您的计算机?
    SAW的LC振荡器(转自www.dwenzhao.cn)
    运动耳机买什么样的好?各类运动蓝牙耳机推荐
    leetcode-每日一题899. 有序队列(思维题)
    NVMe-MI --- Message Servicing Model(消息服务模型)
    软考-高项-论文-信息系统项目的进度管理
  • 原文地址:https://blog.csdn.net/yzq102873/article/details/126240319