文章目录
- 前言
- 一、引出Bean对象的作用域
- 1.普通变量的作用域
- 2.Bean对象的作用域
- 二、Bean对象的作用域
- 1.Bean对象的6种作用域
- 2.设置Bean对象的作用域
- 三、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 有两种设置⽅式:
- 直接设置值:@Scope(“prototype”)
- 使⽤枚举设置:@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对象的作用域和生命周期,是面试中的高频问题,希望大家可以熟练掌握!
路漫漫不止修身,也养性。