• Bean 作用域与生命周期


    目录

    1.案例

    2.作用域

    2.1 Bean 作用域类型

    2.2 设置作用域

    3.Bean 的生命周期

    3.1 Spring 生命周期

    3.2 Bean 的生命周期


    Spring 中 Bean 是最核心的操作,接下来我们来分析 Bean 作用域的问题

    1.案例

    现在有一个公共 Bean(默认是单例模式)的 User 类:

    1. @Component
    2. public class Users {
    3. @Bean(name = "user")
    4. @Scope("prototype") //作用域
    5. public User getUser() {
    6. User user = new User();
    7. user.setId(5);
    8. user.setName("小温");
    9. return user;
    10. }
    11. }

    A 用户进行了修改操作(UserController1 类):

    1. @Controller
    2. public class UserController1 {
    3. @Autowired
    4. private User user;
    5. public void doMethod() {
    6. User user2 = user;
    7. System.out.println("UserController1 修改之前:User -> " + user);
    8. user2.setName("三三");
    9. System.out.println("UserController1 修改之后:User -> " + user);
    10. }
    11. }

    B 用户使用公共 Bean(UserController2 类):

    1. @Controller
    2. public class UserController2 {
    3. @Autowired
    4. private User user;
    5. public void doMethod() {
    6. System.out.println("UserController2 :User -> " + user);
    7. }
    8. }

    打印 A 用户和 B 用户公共 Bean  的值:

    1. public class App {
    2. public static void main(String[] args) {
    3. ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
    4. UserController1 userController1 = context.getBean("userController1", UserController1.class);
    5. userController1.doMethod();
    6. UserController2 userController2 = context.getBean("userController2", UserController2.class);
    7. userController2.doMethod();
    8. }
    9. }

    此时我们发现 B 用户打印的是修改后的值:因为 Bean 默认情况下是单例状态(singleton),也就是所有人使用的都是同⼀个对象,之前我们学单例模式的时候都知道,使用单例可以很大程度上提高性能,所以在 Spring 中 Bean 的作用域默认也是 singleton 单例模式。

    2.作用域

    Bean 的作用域是指 Bean 在 Spring 整个框架中的某种行为模式。 比如 singleton 单例作用域,就表示 Bean 在整个 Spring 中只有⼀份,它是全局共享的,那么当其他人修改了这个值之后,那么另一个人读取到的就是被修改的值。

    2.1 Bean 作用域类型

    1️⃣singleton 单例模式(默认的作用域)

    该作用域下的 Bean 在 IoC 容器中只存在一个实例:获取 Bean(即通过 applicationContext.getBean等方法获取及装配 Bean 都是同⼀个对象

    单例模式的 Bean 是线程安全的吗? 不是(所有人都可以进行操作),使用 ThreaddLocal(本地线程变量)

    2️⃣prototype:原型作用域(多例作用域)

    每次对该作用域下的 Bean 的请求都会创建新的实例:获取Bean(即通过 applicationContext.getBean 等方法获取)及装配 Bean(即通过@Autowired 注入)都是新的对象实例

    3️⃣request:请求作用域

    每次http请求会创建新的Bean实例,类似于prototype;只适用于 Spring MVC 项目(Spring Web)

    4️⃣session:会话作用域

    在⼀个http session中,定义⼀个Bean实例;⽤户回话的共享Bean, 比如:记录⼀个⽤户的登陆信息;只适用于 Spring MVC 项目(Spring Web)

    5️⃣application:应用作用域

    在⼀个http servlet Context中,定义⼀个Bean实例;只适用于 Spring MVC 项目(Spring Web)

    6️⃣websocket:websocket 作用域,只适用于 websocket 作用域(了)

    2.2 设置作用域

    使用 @Scope 标签声明 Bean 的作用域

    1️⃣@Scope("prototype")

    2️⃣@Scope("ConfigurableBeanFactory.SCOPE_PROTOTYPE")

    3.Bean 的生命周期

    3.1 Spring 生命周期

    1. 启动容器
    2. 读取配置进行 Bean 的实例化
    3. 将 Bean 加入到容器中
    4. 装配 Bean 属性(给当前类的属性 DI,进行赋值)
    5. 使用项目,执行 Spring 的相关业务代码
    6. 释放 Bean,销毁 Bean
    7. 关闭容器

    3.2 Bean 的生命周期

    1. 实例化(给 Bean 分配内存空间
    2. 设置 Bean 属性(进行依赖注入,将依赖的 Bean 赋值到当前类的属性上)
    3. Bean 的初始化:1️⃣执行各种通知(如 BeanNameAware的接口方法)2️⃣初始化的前置方法 BeanPostProcessor3️⃣初始化方法 @PostConstruct 4️⃣初始化的后置方法 BeanPostProcessor
    4. 使用 Bean
    5. 销毁 Bean :销毁容器的各种方法,如 @PreDestroy、DisposableBean 接口方法、destroy-method。

    示例:

    1. public class BeanLifeComponent implements BeanNameAware {
    2. @Override
    3. public void setBeanName(String s) {
    4. System.out.println("执行了 BeanNameAware -> " + s);
    5. }
    6. //初始化方法——注解
    7. @PostConstruct
    8. public void doPostConstruct() {
    9. System.out.println("执行了 @PostConstruct");
    10. }
    11. //初始化方法——xml
    12. public void myInit() {
    13. System.out.println("执行了 myinit");
    14. }
    15. @PreDestroy
    16. public void doPreDestroy() {
    17. System.out.println("执行了 @PreDestroy");
    18. }
    19. public void sayHi() {
    20. System.out.println("使用 Bean");
    21. }
    22. }

    xml 配置:

    1. "1.0" encoding="UTF-8"?>
    2. <beans xmlns="http://www.springframework.org/schema/beans"
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    4. xmlns:content="http://www.springframework.org/schema/context"
    5. 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">
    6. <content:component-scan base-package="spring.java.demo">content:component-scan>
    7. <bean id="mybean" class="BeanLifeComponent" init-method="myInit">bean>
    8. beans>

    测试类:

    1. public class BeanLifeTest {
    2. public static void main(String[] args) {
    3. ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
    4. BeanLifeComponent component = context.getBean("mybean", BeanLifeComponent.class);
    5. component.sayHi();
    6. context.close();
    7. }
    8. }

  • 相关阅读:
    竞赛 深度学习+opencv+python实现昆虫识别 -图像识别 昆虫识别
    一键自助建站系统api版系统源码
    Bezier曲线与B-Spline曲线
    SpringBoot多环境开发
    【网络工程】3、交换机原理
    PyTorch笔记 - Convolution卷积运算的原理 (3)
    5G移动性增强关键技术及应用分析
    神经网络开发
    【Arduino】实验使用ESP32单片机根据光线变化控制LED小灯开关(图文)
    一维数组和二维数组的使用(int类型)
  • 原文地址:https://blog.csdn.net/m0_72161237/article/details/133885837