Spring Bean的生命周期总结(包含面试题)

目录

一、Bean的初始化过程

1. 加载Spring Bean

2. 解析Bean的定义

3. Bean属性定义

4.  BeanFactoryPostProcessor 扩展接口 

5. 实例化Bean对象

6. Aware感知

7. 初始化方法

8. 后置处理

9. destroy 销毁 

二、Bean的单例与多例模式 

2.1 单例模式(Singleton)

2.2 多例模式(Prototype)

2.3 案例演示:

2.3.1 单例模式:

2.3.2 多例模式:

2.4 总结

三、关于bean的生命周期面试题

1. 请详细描述Spring框架Bean的生命周期包括哪些阶段?

2. 请详细描述一下Spring Bean的初始化过程

3. Spring Bean的销毁过程是怎样的?

4. Spring Bean的后置处理器是什么?在项目中如何使用它?

5. Spring Bean的生命周期中,哪些方法可以进行自定义操作?

6. 什么是Bean的作用域?Spring中有哪些Bean的作用域?

7. Bean的作用域是如何实现的?请描述其原理。

8. 请解释Spring中的延迟初始化是什么,如何配置延迟初始化?


一、Bean的初始化过程

1. 加载Spring Bean

        通过XML、Java annotation(注解)以及Java Configuration(配置类)等方式加载Spring Bean:这是指在Spring容器启动时,通过配置文件、注解或配置类等方式告诉Spring框架需要创建哪些Bean。

2. 解析Bean的定义

BeanDefinitionReader:这是一个解析器,它将Bean的定义解析成Spring内部的BeanDefinition结构。可以将其理解为将配置文件中的<bean>标签转换为Spring框架内部使用的数据结构。

可理解为:将spring.xml中的 <bean> 标签转换成BeanDefinition结构有点类似于XML解析。

3. Bean属性定义

BeanDefinition:它包含了Bean的各种属性和方法,例如id(标识符)、class(类名)、scope(作用域)、ref(依赖的Bean)等。BeanDefinition实际上是将配置文件中的<bean>标签的定义信息存储到对应的BeanDefinition属性中。

例如:

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

4.  BeanFactoryPostProcessor 扩展接口 

BeanFactoryPostProcessor这是Spring容器的扩展接口,它在Spring容器加载完BeanDefinition之后、Bean实例化之前执行。它可以对Bean的元数据(BeanDefinition)进行加工处理,例如填充或修改BeanDefinition的属性。

5. 实例化Bean对象

BeanFactory:它是Bean的工厂,根据我们的要求生产各种不同的Bean。BeanFactory根据配置文件中的BeanDefinition信息来实例化Bean对象,并进行属性赋值。(根据class属性反射机制实例化对象,反射赋值设置属性)

6. Aware感知

Aware 感知是一种机制,它允许在Bean的生命周期中获取对Spring容器的访问权限。通过实现相应的Aware接口,Bean可以获得与Spring容器的交互能力,以便在特定的生命周期阶段执行自定义操作。

在Spring框架中,有多个Aware接口可供实现,每个接口都提供了不同的功能。以下是一些常见的Aware接口及其作用:

  1. BeanNameAware:通过实现BeanNameAware接口,Bean可以获取自己在Spring容器中的名称。这对于需要知道自己在容器中的标识的Bean非常有用。

  2. ApplicationContextAware:通过实现ApplicationContextAware接口,Bean可以获取对Spring应用上下文的引用。这使得Bean能够在需要时访问Spring容器中的其他Bean或执行其他与容器相关的操作。

  3. BeanFactoryAware:通过实现BeanFactoryAware接口,Bean可以获取对Spring Bean工厂的引用。这使得Bean能够在需要时访问Bean工厂中的其他Bean或执行其他与Bean工厂相关的操作。

        通过实现这些Aware接口,Bean可以在初始化过程中获得对Spring容器的访问权限,并在需要时执行特定的操作。这为开发人员提供了更大的灵活性和控制权,使得他们能够根据自己的需求自定义Bean的行为。

7. 初始化方法

        如果Bean定义中指定了初始化方法,容器会在属性赋值完成后调用该方法进行一些初始化操作。可以使用@PostConstruct注解或者实现InitializingBean接口来指定初始化方法。如果Bean在Spring配置文件中配置了 init-method 属性,则会自动调用其配置的初始化方法。

<bean id="paramAction" class="com.ycxw.beanLife.ParamAction" init-method="init"></bean>

8. 后置处理

BeanPostProcessor:它是一个后置处理器,在Bean对象实例化和依赖注入完成后,在显示调用初始化方法之前或之后添加自定义的逻辑。可以将其类比为AOP中的环绕通知。

注:只有当检测到Bean对象实现了BeanPostProcessor接口时,才会执行Before和After方法。

BeanPostProcessor的执行顺序是:

Before方法 -> 初始化Bean(InitializingBean接口和init-method方法)-> After方法。

9. destroy 销毁 

destroy:这是Bean的销毁阶段,如果这个Bean的Spring配置中配置了destroy-method属性,在Spring容器关闭时调用。在销毁阶段,可以执行一些清理工作,例如释放资源等。

<bean id="paramAction" class="com.ycxw.beanLife.ParamAction" destroy-method="destroy"></bean>

二、Bean的单例与多例模式 

2.1 单例模式(Singleton)

  • 单例模式是指在整个应用程序中只创建一个Bean实例。
  • 当使用单例模式时,Spring容器在首次请求该Bean时创建一个实例,并将其缓存起来。之后的每次请求都会返回同一个实例。
  • 优点:减少了对象创建和销毁的开销,节省了系统资源。
  • 缺点:由于共享同一个实例,可能导致状态信息被多个线程共享,需要注意线程安全问题。

2.2 多例模式(Prototype)

  • 多例模式是指每次请求时都会创建一个新的Bean实例。
  • 当使用多例模式时,Spring容器在每次请求该Bean时都会创建一个新的实例,而不是返回之前创建的实例。
  • 优点:每次请求都会返回一个全新的实例,避免了状态信息共享的问题。
  • 缺点:频繁创建和销毁对象可能导致系统资源消耗增加。

2.3 案例演示:

假设我们有一个简单的Counter类,用于计数:

public class Counter {private int count;public int getCount() {return count;}public void increment() {count++;}
}

我们将使用Spring配置来演示单例和多例模式的区别。

2.3.1 单例模式:

<bean id="counter" class="com.example.Counter" scope="singleton" />
public class SingletonDemo {public static void main(String[] args) {ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");Counter counter1 = context.getBean("counter", Counter.class);Counter counter2 = context.getBean("counter", Counter.class);counter1.increment();System.out.println("Counter 1: " + counter1.getCount()); // 输出:Counter 1: 1System.out.println("Counter 2: " + counter2.getCount()); // 输出:Counter 2: 1}
}

在单例模式下,counter1counter2引用的是同一个实例,因此它们的计数值是相同的。

2.3.2 多例模式:

<bean id="counter" class="com.example.Counter" scope="prototype" />
public class PrototypeDemo {public static void main(String[] args) {ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");Counter counter1 = context.getBean("counter", Counter.class);Counter counter2 = context.getBean("counter", Counter.class);counter1.increment();System.out.println("Counter 1: " + counter1.getCount()); // 输出:Counter 1: 1System.out.println("Counter 2: " + counter2.getCount()); // 输出:Counter 2: 0}
}

在多例模式下,counter1counter2引用的是不同的实例,因此它们的计数值是独立的。

2.4 总结

        单例模式和多例模式各有优缺点,具体使用哪种模式取决于应用程序的需求和设计考虑。在需要共享状态或资源的场景下,可以使用单例模式;在需要独立状态或避免状态共享的场景下,可以使用多例模式。

三、关于bean的生命周期面试题

1. 请详细描述Spring框架Bean的生命周期包括哪些阶段?

在Spring框架中,Bean的生命周期包括以下几个阶段:

  1. 实例化(Instantiation):在这个阶段,Spring会根据配置或注解创建Bean的实例。这可以通过构造函数实例化、工厂方法或者Bean容器中的其他方式来实现。

  2. 属性赋值(Population):在实例化后,Spring会通过依赖注入(Dependency Injection)或者属性赋值的方式,将Bean的属性值设置好。这可以通过setter方法注入或者字段注入来实现。

  3. 初始化(Initialization):在属性赋值完成后,Spring会调用Bean的初始化方法。这个方法可以通过实现InitializingBean接口的afterPropertiesSet()方法,或者在配置文件中通过init-method属性指定。

  4. 使用(In Use):在初始化完成后,Bean进入可用状态,可以被其他对象引用和使用。

  5. 销毁(Destruction):当Bean不再被需要时,Spring会调用Bean的销毁方法进行资源释放和清理工作。这个方法可以通过实现DisposableBean接口的destroy()方法,或者在配置文件中通过destroy-method属性指定。

        Bean的销毁阶段只有在容器关闭时才会触发,或者在使用单例作用域的Bean时,当Bean不再被引用时被垃圾回收器回收。以上是Spring框架中Bean的典型生命周期阶段,每个阶段都提供了相应的扩展点,可以在不同阶段进行自定义操作。

2. 请详细描述一下Spring Bean的初始化过程

👆👆👆 请看本文章第一点内容 !!!👆👆👆

3. Spring Bean的销毁过程是怎样的?

Spring Bean的销毁过程可以分为以下几个步骤:

  1. 关闭容器:当Spring容器关闭时,会触发Bean的销毁过程。这可以通过调用ApplicationContextclose()方法或者销毁容器的生命周期回调来实现。

  2. 调用Bean的销毁方法:在容器关闭时,Spring会调用Bean的销毁方法进行资源释放和清理工作。这个方法可以通过实现DisposableBean接口的destroy()方法,或者在配置文件中通过destroy-method属性指定。

  3. 应用Bean后置处理器:在销毁方法调用之前和之后,Spring会应用注册的Bean后置处理器(BeanPostProcessor)。后置处理器可以对Bean进行额外的处理,例如在Bean销毁前后进行一些自定义逻辑。

        Bean的销毁阶段只有在容器关闭时才会触发,或者在使用单例作用域的Bean时,当Bean不再被引用时被垃圾回收器回收。对于原型作用域的Bean,默认情况下,Spring不会管理其销毁过程,需要开发者手动进行资源释放。

        同时,可以通过实现DisposableBean接口或者在配置文件中指定destroy-method属性来定义Bean的销毁方法。在销毁方法中,可以进行一些资源释放、数据库连接关闭等清理工作,确保系统的正常关闭和资源的释放。

4. Spring Bean的后置处理器是什么?在项目中如何使用它?

Spring Bean后置处理器(BeanPostProcessor)是一种特殊类型的Bean,用于在容器实例化、配置和初始化Bean的过程中进行自定义的处理操作。

在项目中使用Bean后置处理器,可以通过以下步骤:

  1. 创建后置处理器类:实现BeanPostProcessor接口,该接口包含两个方法:postProcessBeforeInitialization()postProcessAfterInitialization()

    import org.springframework.beans.BeansException;
    import org.springframework.beans.factory.config.BeanPostProcessor;public class MyBeanPostProcessor implements BeanPostProcessor {@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {// 在Bean初始化之前进行自定义处理操作return bean;}@Overridepublic Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {// 在Bean初始化之后进行自定义处理操作return bean;}
    }
    
  2. 注册后置处理器:在Spring配置文件中,将自定义的后置处理器注册到容器中。

    <bean class="com.example.MyBeanPostProcessor" />
    
  3. 启动容器:在项目启动时,启动Spring容器,容器会自动识别并应用注册的后置处理器。

    ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
    

通过使用Bean后置处理器,可以在Bean的初始化前后进行自定义操作,例如对Bean进行增强、属性修改、代理等。常见的应用场景包括:

  • 属性值注入前后的自定义处理。
  • 在Bean初始化前后执行一些特定的逻辑。
  • 动态代理Bean的生成。
  • 对特定类型的Bean进行特殊处理等。

        需要注意的是,Bean后置处理器会对容器中的所有Bean实例进行处理,因此在编写后置处理器时需要注意逻辑的准确性和性能的影响。

5. Spring Bean的生命周期中,哪些方法可以进行自定义操作?

在Spring Bean的生命周期中,可以进行自定义操作的方法主要包括以下几个:

  1. 初始化方法(Initialization methods):

    • 在配置文件中通过init-method属性指定的方法。
    • 在Bean类中实现InitializingBean接口,并重写afterPropertiesSet()方法。

    初始化方法用于在Bean实例化后,属性注入完成之后执行一些初始化逻辑,例如初始化资源、建立连接等。您可以根据需要选择其中一种方式来定义初始化方法。

  2. 销毁方法(Destruction methods):

    • 在配置文件中通过destroy-method属性指定的方法。
    • 在Bean类中实现DisposableBean接口,并重写destroy()方法。

    销毁方法用于在容器销毁Bean之前执行一些清理操作,例如释放资源、关闭连接等。您可以根据需要选择其中一种方式来定义销毁方法。

  3. 后置处理器(Post-processors):

    后置处理器可以在Bean的初始化前后对Bean进行自定义处理。postProcessBeforeInitialization()方法在初始化方法调用之前被调用,postProcessAfterInitialization()方法在初始化方法调用之后被调用。通过实现这些方法,您可以对Bean进行额外的操作,例如修改属性值、添加代理等。

  4. 实现BeanPostProcessor接口,并重写postProcessBeforeInitialization()postProcessAfterInitialization()方法。

        除了上述方法外,还可以使用其他扩展机制来进行自定义操作,例如使用@PostConstruct@PreDestroy注解来标记初始化方法和销毁方法,或者通过实现BeanFactoryPostProcessorBeanDefinitionRegistryPostProcessor接口来对Bean的定义进行修改。

        通过自定义这些方法,您可以在Spring Bean的生命周期中添加自己的逻辑,以满足特定的需求和业务场景。

6. 什么是Bean的作用域?Spring中有哪些Bean的作用域?

在Spring中,Bean的作用域(Scope)指的是定义了Bean的生命周期和可见范围的一组规则。作用域决定了在不同的上下文中访问和使用Bean的方式。

Spring框架提供了以下几种常用的Bean作用域:

  1. Singleton(默认):单例作用域。在整个应用程序中,只会创建一个Bean实例,并且所有对该Bean的请求都会返回同一个实例。这是Spring的默认作用域。

  2. Prototype:原型作用域。每次请求Bean时,都会创建一个新的Bean实例。每个实例都有自己的状态,因此在使用时需要注意管理和维护。

  3. Request:请求作用域。每个HTTP请求都会创建一个新的Bean实例,并且该Bean实例仅在当前请求的范围内可见。在同一个请求内的不同组件可以共享同一个实例。

  4. Session:会话作用域。每个HTTP会话(Session)都会创建一个新的Bean实例,并且该Bean实例仅在当前会话的范围内可见。在同一个会话内的不同请求可以共享同一个实例。

  5. Global Session:全局会话作用域。仅在基于portlet的web应用中有效。它类似于Session作用域,但是不同portlet之间可以共享同一个实例。

        除了上述常用的作用域,Spring还提供了其他一些作用域,如WebSocket作用域、Application作用域等,用于满足特定的应用需求。

        要指定Bean的作用域,可以使用@Scope注解或在XML配置文件中进行配置。例如,使用@Scope("prototype")将Bean的作用域设置为原型作用域。

        需要根据具体的业务需求和场景选择合适的Bean作用域。默认情况下,使用单例作用域是最常见和推荐的方式,但在某些情况下,原型作用域或其他作用域可能更适合。

7. Bean的作用域是如何实现的?请描述其原理。

在Spring中,Bean的作用域是通过Bean的创建和管理方式来实现的。不同作用域的Bean会有不同的创建和销毁策略。

  1. Singleton作用域:

    • 当容器启动时,Spring会创建并初始化Singleton作用域的Bean实例,并将其放入容器中的单例缓存池中。
    • 之后,每次请求该Bean时,容器会直接从单例缓存池中返回已经创建好的Bean实例。
    • Singleton作用域的Bean在容器关闭时会被销毁。
  2. Prototype作用域:

    • 当容器接收到对Prototype作用域Bean的请求时,会创建一个新的Bean实例。
    • 容器会对该实例进行初始化和依赖注入,并返回给请求方。
    • 对Prototype作用域的Bean,容器不会进行进一步的管理和跟踪,不负责其生命周期的管理。因此,在使用Prototype作用域的Bean时,需要手动管理其销毁。
  3. Request作用域和Session作用域:

    • 当容器接收到对Request作用域或Session作用域的Bean的请求时,会检查当前的请求或会话中是否存在对应的Bean实例。
    • 如果存在,则直接返回该实例;如果不存在,则创建一个新的Bean实例并放入请求或会话中,以便后续的访问和共享。
    • Request作用域和Session作用域的Bean在请求或会话结束时会被销毁。

        实现Bean作用域的关键在于容器的管理和跟踪机制。容器会根据作用域的不同,采取相应的策略来创建、缓存和销毁Bean实例。对于Singleton作用域的Bean,容器会负责管理其单例缓存池;对于Prototype作用域的Bean,容器只负责创建实例,不负责管理其生命周期;对于Request作用域和Session作用域的Bean,容器会将实例与请求或会话关联起来,以便后续的访问和共享。

        需要注意的是,作用域仅仅决定了Bean实例的创建和可见范围,并不影响Bean定义本身。因此,无论是Singleton作用域还是Prototype作用域,Bean的定义方式都是相同的,只是在创建和管理上有所区别。

8. 请解释Spring中的延迟初始化是什么,如何配置延迟初始化?

在Spring中,延迟初始化(Lazy Initialization)是指在容器启动时不立即创建所有的Bean实例,而是在第一次使用该Bean时才进行创建。这可以有效地减少启动时间和资源消耗,特别是对于那些初始化耗时较长或者很少被使用的Bean。

要配置延迟初始化,可以使用两种方式:

  1. 使用注解配置:

    • 在类级别上使用@Lazy注解,表示该类的Bean实例需要延迟初始化。例如:@Lazy
    • 在XML配置文件中,使用lazy-init="true"属性来配置延迟初始化。例如:
      <bean id="myBean" class="com.example.MyBean" lazy-init="true" />
      
  2. 使用编程方式配置:

    • 在Java配置类中,使用@Lazy注解来标记需要延迟初始化的Bean。例如:@Lazy
    • 在XML配置文件中,使用lazy-init="true"属性来配置延迟初始化。例如:
      <bean id="myBean" class="com.example.MyBean" lazy-init="true" />
      

        无论使用注解还是XML配置,配置了延迟初始化的Bean在容器启动时不会被立即创建。而是在第一次被请求或者被依赖注入时才会进行实例化和初始化。这样可以延迟Bean的创建,节省资源并提高应用程序的启动性能。

        需要注意的是,延迟初始化只适用于Singleton作用域的Bean。对于Prototype作用域的Bean,Spring无法延迟初始化,因为每次请求都需要创建一个新的实例。

        延迟初始化可以根据具体的业务需求和性能考虑来选择是否使用。对于那些启动时间较长或者很少被使用的Bean,延迟初始化可以提高应用程序的启动性能和资源利用率。但对于那些频繁被使用的Bean,延迟初始化可能会导致延迟的开销,需要权衡利弊进行选择。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/72607.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

小航助学GESP_C++一级模拟测试卷第3套(含题库答题软件账号)

需要在线模拟训练的题库账号请点击 小航助学编程在线模拟试卷系统&#xff08;含题库答题软件账号&#xff09;_程序猿下山的博客-CSDN博客 需要在线模拟训练的题库账号请点击 小航助学编程在线模拟试卷系统&#xff08;含题库答题软件账号&#xff09;_程序猿下山的博客-CSD…

05-微信小程序常用组件-表单组件

05-微信小程序常用组件-表单组件 文章目录 表单组件button 按钮案例代码 form 表单案例代码 image 图片支持长按识别的码案例代码 微信小程序包含了六大组件&#xff1a; 视图容器、 基础内容、 导航、 表单、 互动和 导航。这些组件可以通过WXML和WXSS进行布局和样式设…

计算机视觉的应用10-图片中的表格结构识别与提取实战

大家好&#xff0c;我是微学AI&#xff0c;今天给大家介绍一下计算机视觉的应用10-图片中的表格结构识别与提取实战&#xff0c;表格结构识别在信息处理领域中具有广泛应用&#xff0c;但由于表格的多样性和复杂性&#xff0c;以及难以准确解析的布局和格式&#xff0c;传统的方…

C#__使用Type类反射数据的基本用法

// 简单介绍 // 元数据&#xff08;metadata&#xff09;&#xff1a;与程序及其类型有关的数据。 // 反射&#xff1a;一个运行的程序查看本身元数据或其他程序集中的元数据的行为 // Assembly类&#xff1a;允许访问给定程序集的元数据&#xff0c;包含了可以加载和执行程序…

类与对象(上)

类与对象&#xff08;上&#xff09; 一、面向过程和面向对象的区别二、类1、类的引入2、类的定义&#xff08;1&#xff09;类的基本定义&#xff08;2&#xff09;类的成员函数的定义方法 3、类的访问限定符4、封装5、驼峰法命名规则6、类的作用域7、类的实例化&#xff08;1…

TCP服务器实现—多进程版,多线程版,线程池版

目录 前言 1.存在的问题 2.多进程版 3.多线程版 4.线程池版 总结 前言 在上一篇文章中使用TCP协议实现了一个简单的服务器&#xff0c;可以用来服务端和客户端通信&#xff0c;但是之前的服务器存在一个问题&#xff0c;就是当有多个客户端连接服务器的时候&#xff0c;服…

强强联手:Eyeshot 2023.2 集成 CAD Exchanger SDK 3.21 Crack

Lab 中的新绘图模式和网格抽取、Eyeshot 集成以及从 CAD Exchanger 3.21.0 中的 CATIA 导入图形 PMI 通过探索新的绘图模式和可调整的网格抽取&#xff0c;更深入地了解实验室&#xff0c;见证与 devDept Eyeshot 的无缝集成&#xff0c;并直接从 CATIA 导入图形 PMI。 今年 8…

linux——MongoDB服务

目录 一、MongoDB概述 相关概念 特性 应用场景 二、目录结构 三、默认数据库 admin local config 四、数据库操作 库操作 文档操作 五、MongoDB数据库备份 一、备份 mongodump mongoexport 二、恢复 mongorestore mongoimport ​编辑 一、MongoDB概述 MongoDB是…

游乐场vr设备虚拟游乐园vr项目沉浸体验馆

在景区建设一个VR游乐场项目可以为游客提供一种新颖、刺激和沉浸式的游乐体验。提高游客的体验类型&#xff0c;以及景区的类目&#xff0c;从而可以吸引更多的人来体验。 1、市场调研&#xff1a;在决定建设VR游乐场项目之前&#xff0c;需要进行市场调研&#xff0c;了解当地…

【Linux】IO多路转接——poll接口

目录 poll初识 poll函数 poll服务器 poll的优点 poll的缺点 poll初识 poll也是系统提供的一个多路转接接口。 poll系统调用也可以让我们的程序同时监视多个文件描述符上的事件是否就绪&#xff0c;和select的定位是一样的&#xff0c;适用场景也是一样的。 poll函数 po…

(二)Git在公司中团队内合作和跨团队合作和分支操作的全部流程(一篇就够)

&#xff08;一&#xff09;Git连接GitHub的全部流程https://blog.csdn.net/m0_65992672/article/details/132333727 团队内协作 项目经理通过git push将代码推送到远程仓库【也就是git、gitee等代码托管中心】,推完以后组员可以通过git clone克隆下来代码&#xff0c;如果组…

初阶C语言-结构体

&#x1f31e; “少年有梦不至于心动&#xff0c;更要付诸行动。” 今天我们一起学习一下结构体的相关内容&#xff01; 结构体 &#x1f388;1.结构体的声明1.1结构的基础知识1.2结构的声明1.3结构成员的类型1.4结构体变量的定义和初始化 &#x1f388;2.结构体成员的访问2.1结…