【Java】Spring——Bean对象的作用域和生命周期


前言

本人是一个普通程序猿!分享一点自己的见解,如果有错误的地方欢迎各位大佬莅临指导,如果你也对编程感兴趣的话,互关一下,以后互相学习,共同进步。这篇文章能够帮助到你的话,劳请大家点赞转发支持一下!

Spring 是包含了众多工具方法的 IoC 容器 。
而IoC = Inversion of Control 翻译成中文是“控制反转”的意思,
也就是说 Spring 是⼀个“控制反转”的容器 。

而IoC容器控制的是Java程序中对象的生命周期。
即 Bean对象 的生命周期。

所以 Bean对象 作为Spring框架中的核心人物,值得我们去深入研究!
在这里插入图片描述


一、引出Bean对象的作用域

1.普通变量的作用域

  • 限定程序中 变量的可用范围 叫做作用域,或者说在源代码中定义变量的某个区域就叫做作用域。

在这里插入图片描述
上图中 变量 a 的作用域就是test方法。

在这里插入图片描述
上图中 变量 b 的作用域就是在Demo类。


2.Bean对象的作用域

Bean对象,存储到了容器中,那么下面我们就通过代码来研究Bean对象的作用域。

注入公共Bean对象
在这里插入图片描述
向容器中注入注入公共Bean对象 Teacher对象,其中的属性name默认为 “张三”。

在这里插入图片描述
在这里插入图片描述

从容器中取出两个Bean对象,分别注入到两个不同用户UserA与UserB中。

在这里插入图片描述
预估结果为:userA中的teacherA的name变为李四,而userB中的teacherB不变仍为张三,且这两个teacherA != teacherB。
在这里插入图片描述
实际情况与预估情况恰恰相反,真相只有一个请看下方!

得出的结论:
只输出了一次 “Teacher 完成实例化”
—》说明
虽然我取了两个Bean对象,但是Spring 只实例化了一个对象


改变了其中一个Bean对象中的name属性,另一个Bean对象中的name属性也跟着一起改变了
—》说明
这两个Bean对象是同一个对象


teacherA == tercherB 输出为 true
—》说明
引用类型 == 比较的是地址,即teacher 与 teacher1记录同一个地址,即这两个Bean对象是同一个对象。


故而,我们从 Spring容器中取出,同一id的Bean对象,取出的都是同一个。


以上情况与之前文章涉及到的一种设计模式,单例模式类似。
Java 多线程4——wait / notify方法的使用 + 单例模式(饿汉/懒汉)

造成以上情况的原因是因为 Bean对象的作用域!
Bean 默认情况下是单例状态(singleton),也就是所有人的使用的都是同一个对象,之前我们学单例模式的时候都知道,使用单例可以很大程度上提高性能,所以在 Spring 中Bean 的作用域默认也是 singleton 单例模式。


二、Bean对象的作用域

Bean的作用域

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

翻译一下就是:Bean的作用域是指,从Spring容器中取出来的对象,是所有人取出来的都是同一个,还是在某些时刻场景下我重新实例化一个对象给你。

1.Bean对象的6种作用域

Spring 容器在初始化一个 Bean 的实例时,同时会指定该实例的作用域。Spring有 6 种作用域,最后四种是基于 Spring MVC 生效的:
1️⃣singleton:单例作用域
2️⃣prototype:原型作用域(多例作用域)
3️⃣request:请求作用域
4️⃣session:回话作用域
5️⃣application:全局作用域
6️⃣websocket:HTTP WebSocket 作用域
注意!!
后 4 种状态是 Spring MVC 中的值,在普通的 Spring 项⽬中只有前两种。

  • singleton (Bean对象的默认作用域)

说明:该作用域下的Bean在IoC容器中只存在一个实例,获取Bean(即通过applicationContext.getBean等方法获取)及装配Bean(即通过@Autowired注入)都是同一个对象。

场景:通常无状态的Bean使⽤该作用域。无状态表示Bean对象的属性状态不需要更新。

  • prototype

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

场景:通常有状态的Bean使用该作用域。

  • request (限定SpringMVC中使用)

说明:每次http请求会创建新的Bean实例,类似于prototype。

场景:⼀次http的请求和响应的共享Bean。

  • session (限定SpringMVC中使用)

说明:在⼀个http session中,定义一个Bean实例

场景:用户回话的共享Bean, 比如:记录⼀个用户的登陆信息。

  • application (限定SpringMVC中使用)

说明:在⼀个http servlet Context中,定义一个Bean实例

场景:Web应用的上下文信息,比如:记录一个应用的共享信息。

  • websocket (限定SpringMVC中使用)

说明:在一个HTTP WebSocket的生命周期中,定义⼀个Bean实例。

场景:WebSocket的每次会话中,保存了⼀个Map结构的头信息,将用来包裹客户端消息头。第⼀次初始化后,直到WebSocket结束都是同⼀个Bean。


2.设置Bean对象的作用域

使用@Scope 注解 就可以来声明 Bean对象的作用域。

@Scope 标签既可以修饰方法也可以修饰类,@Scope 有两种设置⽅式:

  1. 直接设置值:@Scope(“prototype”)
  2. 使⽤枚举设置:@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)

就以修改上面不符合预估情况的代码为例,来使用@Scope 注解。

在这里插入图片描述
在这里插入图片描述

此时就符合预估情况了。


三、Bean对象的生命周期

所谓的生命周期指的是一个对象从诞生到销毁的整个生命过程,我们把这个过程就叫做一个对象的生命周期。
Bean 的生命周期分为以下 5 大部分:

1️⃣实例化 Bean(为 Bean 分配内存空间)。
2️⃣设置属性(Bean 注入和装配)。
3️⃣Bean 初始化 {

  • 实现了各种 Aware 通知的方法,如 BeanNameAware、BeanFactoryAware、ApplicationContextAware 的接口方法。
  • 执行 BeanPostProcessor 初始化前置方法。
  • 执行 @PostConstruct 初始化方法,依赖注入操作之后被执行。
  • 执行自己指定的 init-method 方法(如果有指定的话)。
  • 执行 BeanPostProcessor 初始化后置方法。
    }

4️⃣使用 Bean。
5️⃣销毁 Bean,销毁容器的各种方法,如 @PreDestroy、DisposableBean 接口方法、destroy-method。

在这里插入图片描述

注意!!!

第二步的设置属性,是为类中由Spring容器自动注入Bean对象的变量,注入Bean对象。
比如下面的UserA也会注入到容器中,而他其中又含有其他的被注入到容器中的Bean对象,故而在实例化UserA类的时候要将Teacher这个Bean对象注入。

在这里插入图片描述

所以这也就注定了,第二步设置属性(Bean 注入和装配),与第三步Bean 初始化不能调换位置,
因为第三步的各种方法中可能会使用到类中的其他Bean对象,所以要先包装类中其他的Bean对象都装配完成了。


总结

Bean对象的作用域和生命周期,是面试中的高频问题,希望大家可以熟练掌握!

路漫漫不止修身,也养性。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值