提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
本文主要记录spring的单例bean、切面编程AOP、spring事务、循环依赖、常见注解等
提示:以下是本篇文章正文内容,下面案例可供参考



Spring框架中的bean是单例的

singleton :bean在每个Spring IOC容器中只有一个实例。
prototype:一个bean的定义可以有多个实例。
count是可修改的成员变量需要考虑线程安全
userService是无法修改的成员变量不需要考虑线程安全




AOP称为面向切面编程,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为“切面”(Aspect),减少系统中的重复代码,降低模块间的耦合度,同时提高了系统的可维护性。
常见的AOP使用场景:
1.记录操作日志(每一个service记录操作)
2.缓存处理(业务加缓存,写在AOP的切面拦截需要添加缓存的业务方法)
3.Spring中内置的事务处理
获取请求的用户名、请求方式、访问地址、模块名称、登录ip、操作时间,记录到数据库的日志表中,如下图所示:

后台有四个请求接口,需要记录操作日志,使用AOP提供的环绕通知做一个切面,相当于一个通过的代码,遇到请求的时候就可以执行这段代码。

代码解释:https://zhuanlan.zhihu.com/p/596205936?utm_id=0
Spring支持编程式事务管理和声明式事务管理两种方式。



对spring框架的深入理解、复杂业务的编码经验
转账的代码中出现一个异常,由于这个异常会被try-catch捕获却没有抛出,所以可能导致事务失效(如果异常没有被捕获,就会发生事务回滚)

原因分析: 事务通知只有捉到了目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理掉异常,事务通知无法知悉
解决方法: 在catch块添加throw new RuntimeException(e)抛出

系统找不到指定文件,抛出一个检查异常

原因分析: Spring 默认只会回滚非检查异常
解决方法: 配置rollbackFor属性Transactional(rollbackFor=Exception.class),一旦出现异常都会被事务控制

方法没有用public

原因分析: Spring为方法创建代理、添加事务通知、前提条件都是该方法是public的
解决方法: 方法改为public


Spring容器是如何管理和创建bean实例
方便调试和解决问题
Spring容器在进行实例化时,会将xml配置的的信息封装成一个BeanDefinition对象,Spring根据BeanDefinition来创建Bean对象,里面有很多的属性用来描述Bean







在创建A对象的同时需要使用的B对象,在创建B对象的同时需要使用到A对象



Spring解决循环依赖是通过三级缓存,对应的三级缓存如下所示:

如果要想打破循环依赖,就需要一个中间人的参与,这个中间人就是二级缓存。
一级和二级缓存可以解决一般对象的循环依赖

代理对象需要用到三级缓存

构造方法注入产生的循环依赖问题,使用延迟加载解决问题




Springmvc的执行流程是这个框架最核心的内容
重要的组件:前端控制器、处理器映射器、处理器适配器、视图解析器





自动配置主要依赖于@SpringBootApplication注解,其中还包含了三个注解

@SpringBootConfiguration:该注解与@Configuration注解作用相同,用来声明当前也是
个配置类。
@ComponentScan:组件扫描,默认扫描当前引导类所在包及其子包。
@EnableAutoConfiguration: SpringBoot实现自动化配置的核心注解。








查询用户信息的同时也可以查询到相关订单信息
UserMapper:

OrderMapper:

UserTest.java 打印输出用户信息

执行结果:

fetchType="lazy" 开启局部延迟加载
UserMapper:

UserTest.java 打印输出用户信息

执行结果:先调用查询用户信息,然后因为调用了getOrderList(),所以还执行了查找对应订单的sql

在配置文件中开启全局延迟加载:

延迟加载的原理



● 本地缓存:基于PerpetualCache,本质是一个HashMap
● 一级缓存:作用域是session级别
● 二级缓存:作用域是namespace和mapper的作用域,不依赖于session

一级缓存:基于PerpetualCache的 HashMap本地缓存,其存储作用域为Session,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存

因为这两个都是查询的同一个id的用户信息,所以第二次查询的时候直接在缓存中取,而不需要再次执行SQL语句

二级缓存是基于namespace和mapper的作用域起作用的,不是依赖于SQL session,默认也是采用PerpetualCache,HashMap存储(默认关闭)


开启二级缓存的两个步骤:
全局配置文件

映射文件:使用标签让当前mapper生效二级缓存

执行结果:只调用了一次SQL

注意事项:

https://www.bilibili.com/video/BV1yT411H7YK?p=35&vd_source=98092b0aee05ae7c890b09fe07f13df4