【JavaEE】Spring的开发要点总结(4)
文章目录
- 【JavaEE】Spring的开发要点总结(4)
- 1. Bean的作用域
- 1.1 一个例子感受作用域的存在
- 1.2 通过例子说明作用域的定义
- 1.3 六种不同的作用域
- 1.3.1 singleton单例模式(默认作用域)
- 1.3.2 prototype原型模式
- 1.3.3 request请求作用域
- 1.3.4 session会话作用域
- 1.3.5 application全局/应用作用域
- 1.3.6 "websocket" HTTP WebSocket作用域
- 1.4 设置Bean的作用域
- 2. Bean的生命周期
- 2.1 Spring的执行流程
- 2.2 Spring 的生命周期
- 2.3 Bean的生命周期
- 2.3.1 Bean初始化
- 2.3.2 Bean生命周期代码演示
- 2.3.3 为什么属性设置比Bean初始化早
在学习Spring中,Bean是最核心的操作资源
在学习C语言或者JavaSE的时候,熟悉一个变量的作用域非常重要,否则会出现很多错误,并且违背一些设计上的初心~
知己知彼,才能百战百胜~
之前的代码,我们只是一个简单的读操作,没有涉及其他,所以作用域感受不明显,接下来一个例子说明一下~
背景故事:
一个公司开发了一个外卖平台,这个公司打算将这个平台卖给别的公司,赚收成和维护费,但是每个公司都有特定的要求,而员工三人(小马、大马、老马),负责这个项目,小马负责公司原本的代码,大马负责A公司的外卖平台的代码,老马则负责B公司。
大马和老马的工作就是,对一些功能进行删减,添加个别的功能~
三人各自完成各自的业务~
所以就会如下的项目结构(实际情况要比这复杂很多,这个例子只是为了演示罢了):
这个Users类,就是一些原始的User的诞生和定义的地方~
那么可能就会有以下场景:
再每次设置后都打印一次:
System.out.println(user2);
他们原本的意思就是,他们从spring中获取一个Bean对象(用户),设置对应的属性,为自己所用~
现在我们来测试一下:
@Component
public class Test {
@Autowired
private UserController userController;
@Autowired
private UserController1 userController1;
@Autowired
private UserController2 userController2;
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
Test test = context.getBean("test", Test.class);
test.userController.doMethod();
test.userController1.doMethod();
test.userController2.doMethod();
}
}
从结果可以看出,下一次的更新,是在前一次更新的基础上进行的!
所以可以说明,他们用的Bean,是同一个!就像C语言全局变量那样~
这就是作用域中的一种:单例Bean对象
改变Bean对象的作用域也很简单,只需要一个注解@Scope(意思就是作用域)~
默认情况下就是:singleton
我们如果要让小马大马老马获得的Bean对象都不一样,可以设置为:
prototype(原型/多例)
Bean的作用域指的就是Bean对象在Spring整个框架中的某种行为模式:
Spring普通项目( Spring Core)其实就前面两种:singleton 和 prototype
后四种值则是在 Spring MVC 项目中的值
单例模式的效率比较高(性能好)
经典的面试题:单例模式的Bean是线程安全的吗?
不是线程安全的~
解决方案:使用ThreadLocal(本地线程变量)
文章推荐:ThreadLocal不好用?那是你没用对!| Java Debug 笔记 - 掘金 (juejin.cn)
ThreadLocal的基础方法:
可能存太多没有remove,内存溢出的问题也会出现,感兴趣的可以去了解一下
相比于使用锁来解决线程安全问题
- 使用ThreadLocal可能会导致结果与锁不一致的情况
- 特别是在多个线程之间存在依赖关系的情况下。
- 因为每个线程都有自己的数据副本,如果线程之间需要共享数据并进行协作,那么就需要额外的协调机制来保证数据的一致性。
- 否则,可能会出现一个线程修改了数据,但其他线程并不知道的情况,导致结果不一致。
所以在使用ThreadLocal解决线程安全问题时,**需要根据具体的业务场景来评估是否适合使用ThreadLocal,**并确保线程之间的数据协作和一致性。
对于一些依赖全局状态的场景,使用锁可能更适合。
每次获取(DI)的都是一个原型的对象:
顾名思义,在每一次HTTP请求的时候,创建一次原型,与prototype类似
在一次HTTP请求和响应中,共享Bean
注意:限定在Spring MVC中使用
因为Spring Core项目不支持HTTP
Spring MVC项目也叫作 Spring Web项目,支持HTTP
顾名思义,一个HTTP Session中,共享Bean
注意:限定在Spring MVC中使用
后面四种可能会比较难理解,这是因为我们还没有接触Spring MVC,所以不太了解具体用法!
在一个http servlet Context中,共享Bean
即一个Context容器,共享Bean
注意:限定在Spring MVC中使用
对于普通的Spring项目是不能用这个值的
但是对于singleton单例模式,Bean的作用域不超过一个ApplicationContext对象(一个context是一个容器,不同context进行各自的注入…):
singleton和application有什么区别呢?
- 前者是Spring Core的全局作用域,作用于IoC容器
- 后者是Spring MVC(Spring Web)的全局作用域,作用于Servlet容器
了解即可
在一个HTTP WebSocket的生命周期中,共享Bean
就是一个特殊项目里使用的特殊值罢了
注意:限定在Spring WebSocket中使用
WebSocket的每次会话中,保存了一个Map结构的头信息,将用来包裹客户端消息头。第一次初始化后,直到WebSocket结束都是同一个Bean。
如果对WebSocket项目感兴趣的同学可以去学习,如果不感兴趣,了解一下也可以
效果一致~
对于 & :了解即可!
笼统的流程:
跟流程基本一致(粗糙的了解):
所谓的生命周期指的是一个对象从诞生到销毁的整个生命周期,我们把这个过程叫做一个对象的生命周期
Spring的一生其实也差不多是Bean 的一生吧~
Bean的生命周期,也是经典的面试题!
Bean的生命周期分为以下5大部分:
如图是xml的方式去决定使用什么初始化方法
用注解设置多个初始化方法也更加方便~
对于初始化和销毁方法的设置,还有很多其他的方法!
测试:
用子类,有更多的方法
获取Bean,使用Bean
扫描路径不要删掉,即使没用也要设置的
效果:
注解的优先级比较高~
顺序正如我们所料~
其实这个很容易想,例如一下操作:
如果user这个没有指向一块内存空间,只是null,那么就会空指针异常~
而实例化和属性注入之后,相当于在这里放了个箱子,之后的操作有了对象
文章到此结束!谢谢观看
可以叫我 小马,我可能写的不好或者有错误,但是一起加油鸭!Spring Core普通项目的讲解告一段落,接下来是Spring Boot的学习,敬请期待!
代码位置:SpringDemo4/src/main/java · 游离态/马拉圈2023年8月 - 码云 - 开源中国 (gitee.com)