【Spring】Bean的生命周期与单例实现(二)

简介: 【Spring】Bean的生命周期与单例实现(二)

五、面试题讲解

5.1.Bean的生命周期

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

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

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

<bean id="" class="" scope=""> -----> BeanDefinition(id/class/scope)

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

注意:

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

在bean实例化之前执行的

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

属性填充、修改等操作

5.BeanFactory:bean工厂。它按照我们的要求生产我们需要的各种各样的bean。

例如:

BeanFactory -> List<BeanDefinition>

BeanDefinition(id/class/scope/init-method)

<bean class="com.zking.spring02.biz.BookBizImpl"/>

foreach(BeanDefinition bean : List<BeanDefinition>){

  //根据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

8.destory:销毁

结论:

1.xml/annotation/configuation 配置JavaBean。

2.BeanDefinitionReader解析配置的Javabean得到BeanDefinition,最终得到一个List集合。

3.触发BeanFactoryPostProcessor,JavaBean初始化之前执行自己的业务。

4.spring中的beanFactory,会通过List<BeanDefinition>集合遍历初始化所有的JavaBean对象(反射实例化)。

5.如果我们自己的JavaBean需要调动spring上下文中的资源,那么需要实现*Aware感知接口

6.如果自己的JavaBean已经初始化好了,还需要做扩展,那么需要借助BeanPostProcessor来实现。

7.销毁(destory)。

5.2.Spring中JavaBean是单例还是多例

1.默认是单例的,但是可以配置多例

2.单例的优点,节约内存,弊端是有变量污染(多例的优点是无变量污染,但是极其消耗内存)

一个简短风趣的故事来表达单例和多例的区别以及各自的优缺点。


在一个奇幻的魔法学校里,有一位特殊的老师叫做“魔力曼”。他拥有超强的魔法能力,并且负责教授学生们各种神奇的法术。

对于魔力曼来说,他是一个不折不扣的单例。整个学校只有他一个人具备如此强大的魔法力量,每个学生和教职员工都会向他寻求帮助和指导。他总是被围绕在学校的中心,众人都追捧他的存在。

然而,由于魔力曼的特殊地位,他的魔法力量渗透到了整个学校的环境中。无论是在课堂上还是在学校其他地方,都充满了他独特的魔法气息。有时候,当学生们试图施展自己的法术时,结果却受到了魔力曼的力量影响,产生了意想不到的效果。

相比之下,还有一位老师叫做“多地云”。她是一个多例,每个班级都能找到她的身影。她拥有温和而灵活的魔法力量,善于根据学生的需求和情况进行调整。

当学生们在多地云的指导下施展法术时,她会根据不同的班级环境和需求,帮助他们发挥出最大的潜力。由于她的魔法力量不会污染其他班级的影响,所以学生们能够更加自由地探索和发展自己的魔法技能。

这个故事告诉我们,单例和多例各有优缺点。单例的优势在于集中管理和权威性,但也容易产生变量污染的问题。而多例则可以为不同场景提供定制化的服务,避免了变量污染的潜在问题。

在设计软件时,我们应该根据具体需求选择适合的模式。对于需要全局统一访问和共享资源的情况,可以使用单例。而对于需要灵活、无变量污染的场景,多例会是更好的选择。我们需要平衡单例和多例的优缺点,并根据具体需求做出明智的选择。

论证:

ParamAction.java

package com.csdn.xw.aop.beanLife;
import java.util.List;
public class ParamAction {
  private int age;
  private String name;
  private List<String> hobby;
  private int num = 1;
  // private UserBiz userBiz = new UserBizImpl1();
  public ParamAction() {
    super();
  }
  public ParamAction(int age, String name, List<String> hobby) {
    super();
    this.age = age;
    this.name = name;
    this.hobby = hobby;
  }
  public void execute() {
    // userBiz.upload();
    // userBiz = new UserBizImpl2();
    System.out.println("this.num=" + this.num++);
    System.out.println(this.name);
    System.out.println(this.age);
    System.out.println(this.hobby);
  }
}

spring-context.xml

  <bean id="paramAction" class="com.csdn.xw.aop.beanLife.ParamAction">
        <constructor-arg name="name" value="三丰"></constructor-arg>
        <constructor-arg name="age" value="21"></constructor-arg>
        <constructor-arg name="hobby">
            <list>
                <value>抽烟</value>
                <value>烫头</value>
                <value>大保健</value>
            </list>
        </constructor-arg>
    </bean>

Test测试类

package com.csdn.xw.aop.beanLife;
import org.junit.Test;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;
/*
 * spring bean的生命週期
 * spring bean的單例多例
 */
public class Demo2 {
  // 体现单例与多例的区别
  @Test
  public void test1() {
    ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
//    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
    ParamAction p1 = (ParamAction) applicationContext.getBean("paramAction");
    ParamAction p2 = (ParamAction) applicationContext.getBean("paramAction");
    // System.out.println(p1==p2);
    p1.execute();
    p2.execute();
//    单例时,容器销毁instanceFactory对象也销毁;多例时,容器销毁对象不一定销毁;
    applicationContext.close();
  }
  // 体现单例与多例的初始化的时间点 instanceFactory
  @Test
  public void test2() {
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
  }
  // BeanFactory会初始化bean对象,但会根据不同的实现子类采取不同的初始化方式
  // 默认情况下bean的初始化,单例模式立马会执行,但是此时XmlBeanFactory作为子类,单例模式下容器创建,bean依赖没有初始化,只有要获取使用bean对象才进行初始化
  @Test
  public void test3() {
    // ClassPathXmlApplicationContext applicationContext = new
    // ClassPathXmlApplicationContext("/spring-context.xml");
    Resource resource = new ClassPathResource("/spring-context.xml");
    BeanFactory beanFactory = new XmlBeanFactory(resource);
//    InstanceFactory i1 = (InstanceFactory) beanFactory.getBean("instanceFactory");
  }
}

运行结果:

这一次我们手动设置为多例scope="prototype"再看看

5.3.单例的JavaBean和多例JavaBean是什么的时候才会创建

单例:JavaBean跟着Spring上下文初始化,容器生对象生,容器死对象死

多例:JavaBean是使用的时候才会创建,销毁跟着JVM走

论证:

InstanceFactory.java

package com.csdn.xw.aop.beanLife;
public class InstanceFactory {
  public void init() {
    System.out.println("初始化方法");
  }
  public void destroy() {
    System.out.println("销毁方法");
  }
  public void service() {
    System.out.println("业务方法");
  }
}

spring-context.xml

  <bean id="instanceFactory" class="com.csdn.xw.aop.beanLife.InstanceFactory"
          scope="prototype" init-method="init" destroy-method="destroy"></bean>

可以看到上面的xml配置的是多例的,我们调用一下test2看我们的"Bean初始化"了没有?

可以看到并没有,因为我们只是获取spring的上下文对象而没有去获取Bean对象,现在我们换成单例scope="singleton"来看看结果。

这时候我们发现不管我们有无获取Bean对象,都已经创建JavaBean了。

总结:为什么单例是容器创建完了它就初始化了呢?因为单例的思想是不管你什么时候用都要去创建,要是等你浏览器发个请求,再去创建是非常降低用户的体验感,反正只会创建一次何不在你启动项目的时候就创建好了,把时间的消耗放到了启动项目上。如果是多例,有一百个你就创建一百个吗?一千个你也创建一千个,万一我一千个我只使用一个呢?剩下的九百九十九个就是浪费了,所以,多例只会在你使用的时候创建。

5.4.我们使用单例一定会初始化JavaBean吗

BeanFactory会初始化bean对象,但会根据不同的实现子类采取不同的初始化方式,默认情况下bean的初始化,单例模式立马会执行,但是此时XmlBeanFactory作为子类,单例模式下容器创建,bean依赖没有初始化,只有要获取使用bean对象才进行初始化。

论证:

调用test3查看一下结果

XmlBeanFactory作为子类,单例模式下容器创建,bean依赖没有初始化,那我们获取并使用一下

这时候Bean就被创建了。

到这里我的分享就结束了,欢迎到评论区探讨交流!!

如果觉得有用的话还请点个赞吧 ♥  ♥

相关文章
|
5天前
|
安全 Java Spring
Spring框架中的单例Bean是线程安全的吗?
Spring框架中的单例Bean是线程安全的吗?
10 1
|
5天前
|
XML 前端开发 Java
【JavaEE】深入了解Spring中Bean的可见范围(作用域)以及前世今生(生命周期)
【JavaEE】深入了解Spring中Bean的可见范围(作用域)以及前世今生(生命周期)
8 0
|
5天前
|
存储 缓存 Java
【JavaEE】Spring中注解的方式去获取Bean对象
【JavaEE】Spring中注解的方式去获取Bean对象
3 0
|
5天前
|
存储 Java 对象存储
【JavaEE】Spring中注解的方式去存储Bean对象
【JavaEE】Spring中注解的方式去存储Bean对象
9 0
|
5天前
|
存储 Java 对象存储
【JavaEE】DI与DL的介绍-Spring项目的创建-Bean对象的存储与获取
【JavaEE】DI与DL的介绍-Spring项目的创建-Bean对象的存储与获取
9 0
|
5天前
|
消息中间件 安全 Java
在Spring Bean中,如何通过Java配置类定义Bean?
【4月更文挑战第30天】在Spring Bean中,如何通过Java配置类定义Bean?
22 1
|
5天前
|
Java 应用服务中间件 Maven
SpringBoot 项目瘦身指南
SpringBoot 项目瘦身指南
63 0
|
5天前
|
缓存 安全 Java
Spring Boot 面试题及答案整理,最新面试题
Spring Boot 面试题及答案整理,最新面试题
142 0
|
5天前
|
存储 JSON Java
SpringBoot集成AOP实现每个接口请求参数和返回参数并记录每个接口请求时间
SpringBoot集成AOP实现每个接口请求参数和返回参数并记录每个接口请求时间
49 2
|
5天前
|
前端开发 搜索推荐 Java
【Spring底层原理高级进阶】基于Spring Boot和Spring WebFlux的实时推荐系统的核心:响应式编程与 WebFlux 的颠覆性变革
【Spring底层原理高级进阶】基于Spring Boot和Spring WebFlux的实时推荐系统的核心:响应式编程与 WebFlux 的颠覆性变革