一文带你认识Spring事务
前言
只有光头才能变强。
文本已收录至我的GitHub仓库,欢迎Star: ZhongFuCheng3y/3y
Spring事务管理我相信大家都用得很多,但可能仅仅局限于一个@Transactional
注解或者者在XML
中配置事务相关的东西。不论怎样说,日常可能足够我们去用了。但作为程序员,无论是为了面试还是说更好把控自己写的代码,还是应该得多多理解一下Spring事务的少量细节。
这里我抛出几个问题,看大家能不能瞬间答得上:
- 假如嵌套调用含有事务的方法,在Spring事务管理中,这属于哪个知识点?
- 我们使用的框架可能是
Hibernate/JPA
或者者是Mybatis
,都知道的底层是需要一个session/connection
对象来帮我们执行操作的。要保证事务的完整性,我们需要多组数据库操作要使用同一个session/connection
对象,而我们又知道Spring IOC所管理的对象默认都是单例的,这为啥我们在使用的时候不会引发线程安全问题呢?内部Spring究竟干了什么? - 人家所说的BPP又是啥东西?
- Spring事务管理重要接口有哪几个?
一、阅读本文需要的基础知识
阅读这篇文章的同学我默认大家都对Spring事务相关知识有肯定的理解了。(ps:假如不理解点解具体的文章去阅读再回到这里来哦)
我们都知道,Spring事务是Spring AOP的最佳实践之一,所以说AOP入门基础知识(简单配置,使用)是需要先知道的。假如想更加全面理解AOP可以看这篇文章:AOP重要知识点(术语详情、全面使用)。说到AOP就不能不说AOP底层原理:动态代理商设计模式。到这里,对AOP已经有一个基础的认识了。于是我们即可以使用XML/注解方式来配置Spring事务管理。
在IOC学习中,可以知道的是Spring中Bean的生命周期(引出BPP对象)并且IOC所管理的对象默认都是单例的:单例设计模式,单例对象假如有”状态“(有成员变量),那么多线程访问这个单例对象,可能就造成线程不安全。那么何为线程安全?,处理线程安全有很多方式,但其中有一种:让每一个线程都拥有自己的一个变量:ThreadLocal
假如对我以上说的知识点不太理解的话,建议点击蓝字进去学习一番。
二、两个不靠谱直觉的例子
2.1第一个例子
之前朋友问了我一个例子:
在Service层抛出Exception,在Controller层捕获,那假如在Service中有异常,那会事务回滚吗?
// Service方法 @Transactionalpublic Employee addEmployee() throws Exception { Employee employee = new Employee("3y", 23); employeeRepository.save(employee); // 假设这里出了Exception int i = 1 / 0; return employee;}// Controller调用@RequestMapping("/add")public Employee addEmployee() { Employee employee = null; try { employee = employeeService.addEmployee(); } catch (Exception e) { e.printStackTrace(); } return employee;}
我第一反应:不会回滚吧。
- 我当时是这样想的:由于Service层已经抛出了异常,由Controller捕获。那能否回滚应该由Controller的catch代码块中逻辑来决定,假如catch代码块没有回滚,那应该是不会回滚。
但朋友经过测试说,可以回滚阿。(pappapa打脸)
发生了运行时Exception,Spring事务管理自动回滚
看了一下文档,原来文档有说明:
By default checked exceptions do not result in the transactional interceptor marking the transaction for rollback and instances of RuntimeException and its subclasses do
结论:假如是编译时异常不会自动回滚,假如是运行时异常,那会自动回滚!
2.2第二个例子
第二个例子来源于知乎@柳树文章,文末会给出相应的URL
我们都知道,带有@Transactional
注解所包围的方法就能被Spring事务管理起来,那假如我在当前类下使用一个没有事务的方法去调用一个有事务的方法,那我们这次调用会怎样样?能否会有事务呢?
用代码来形容一下:
// 没有事务的方法去调用有事务的方法public Employee addEmployee2Controller() throws Exception { return this.addEmployee();}@Transactionalpublic Employee addEmployee() throws Exception { employeeRepository.deleteAll(); Employee employee = new Employee("3y", 23); // 模拟异常 int i = 1 / 0; return employee;}
我第一直觉是:这跟Spring事务的传播机制有关吧。
其实这跟Spring事务的传播机制没有关系,下面我讲述一下:
- Spring事务管理用的是AOP,AOP底层用的是动态代理商。所以假如我们在类或者者方法上标注注解
@Transactional
,那么会生成一个代理商对象。
接下来我用图来说明一下:
Spring会自动生成代理商对象
显然地,我们拿到的是代理商(Proxy)对象,调用addEmployee2Controller()
方法,而addEmployee2Controller()
方法的逻辑是target.addEmployee()
,调用回原始对象(target)的addEmployee()
。所以这次的调用压根就没有事务存在,更谈不上说Spring事务传播机制了。
原有的数据:
原有的数据
测试结果:压根就没有事务的存在
没有事务的存在
2.2.1再延伸一下
从上面的测试我们可以发现:假如是在本类中没有事务的方法来调用标注注解@Transactional
方法,最后的结论是没有事务的。那假如我将这个标注注解的方法移到别的Service对象上,有没有事务?
@Servicepublic class TestService { @Autowired private EmployeeRepository employeeRepository; @Transactional public Employee addEmployee() throws Exception { employeeRepository.deleteAll(); Employee employee = new Employee("3y", 23); // 模拟异常 int i = 1 / 0; return employee; }}@Servicepublic class EmployeeService { @Autowired private TestService testService; // 没有事务的方法去调用别的类有事务的方法 public Employee addEmployee2Controller() throws Exception { return testService.addEmployee(); }}
测试结果:
抛出了运行时异常,但我们的数据还是存在的!
由于我们用的是代理商对象(Proxy)去调用addEmployee()
方法,那就当然有事务了。
看完这两个例子,有没有觉得3y的直觉是真的水!
三、Spring事务传播机制
假如嵌套调用含有事务的方法,在Spring事务管理中,这属于哪个知识点?
在当前含有事务方法内部调用其余的方法(无论该方法能否含有事务),这就属于Spring事务传播机制的知识点范畴了。
Spring事务基于Spring AOP,Spring AOP底层用的动态代理商,动态代理商有两种方式:
- 基于接口代理商(JDK代理商)
- 基于接口代理商,但凡类的方法非public修饰,或者者用了static关键字修饰,那这些方法都不能被Spring AOP加强
- 基于CGLib代理商(子类代理商)
- 基于子类代理商,但凡类的方法使用了private、static、final修饰,那这些方法都不能被Spring AOP加强
至于为啥以上的情况不能加强,用你们的脑瓜子想一下就知道了。
值得说明的是:那些不能被Spring AOP加强的方法并不是不能在事务环境下工作了。只需它们被外层的事务方法调用了,因为Spring事务管理的传播级别,内部方法也可以工作在外部方法所启动的事务上下文中。
至于Spring事务传播机制的几个级别,我在这里就不贴出来了。这里只是再次解释“啥情况才是属于Spring事务传播机制的范畴”。
四、多线程问题
我们使用的框架可能是
Hibernate/JPA
或者者是Mybatis
,都知道的底层是需要一个session/connection
对象来帮我们执行操作的。要保证事务的完整性,我们需要多组数据库操作要使用同一个session/connection
对象,而我们又知道Spring IOC所管理的对象默认都是单例的,这为啥我们在使用的时候不会引发线程安全问题呢?内部Spring究竟干了什么?
回想一下当年我们学Mybaits的时候,是怎样编写Session工具类?
Mybatis工具类部分代码截图
没错,用的就是ThreadLocal,同样地,Spring也是用的ThreadLocal。
以下内容来源《精通 Spring4.x》
我们知道在一般情况下,只有无状态的Bean才可以在多线程环境下共享,在Spring中,绝大部分Bean都可以公告为singleton作用域。就是由于Spring对少量Bean(如RequestContextHolder、TransactionSynchronizationManager、LocaleContextHolder等)中非线程安全状态的“状态性对象”采用ThreadLocal封装,让它们也成为线程安全的“状态性对象”,因而,有状态的Bean就能够以singleton的方式在多线程中工作。
我们可以试着点一下进去TransactionSynchronizationManager中看一下:
全都是ThreadLocal
五、啥是BPP?
BBP的全称叫做:BeanPostProcessor,一般我们俗称对象后解决器
- 简单来说,通过BeanPostProcessor可以对我们的对象进行“加工解决”。
Spring管理Bean(或者者说Bean的生命周期)也是一个常考的知识点,我在秋招也重新整理了一下步骤,由于比较重要,所以还是在这里贴一下吧:
- ResouceLoader加载配置信息
- BeanDefintionReader解析配置信息,生成一个一个的BeanDefintion
- BeanDefintion由BeanDefintionRegistry管理起来
- BeanFactoryPostProcessor对配置信息进行加工(也就是解决配置的信息,一般通过PropertyPlaceholderConfigurer来实现)
- 实例化Bean
- 假如该Bean
配置/实现
了InstantiationAwareBean,则调用对应的方法 - 使用BeanWarpper来完成对象之间的属性配置(依赖)
- 假如该Bean
配置/实现了
Aware接口,则调用对应的方法 - 假如该Bean配置了BeanPostProcessor的before方法,则调用
- 假如该Bean配置了
init-method
或者者实现InstantiationBean,则调用对应的方法 - 假如该Bean配置了BeanPostProcessor的after方法,则调用
- 将对象放入到HashMap中
- 最后假如配置了destroy或者者DisposableBean的方法,则执行销毁操作
Application中Bean的公告周期
其中也有关于BPP图片:
BBP所在的位置
5.1为什么特意讲BPP?
Spring AOP编程底层通过的是动态代理商技术,在调用的时候一定用的是代理商对象。那么Spring是怎样做的呢?
我只要要写一个BPP,在postProcessBeforeInitialization或者者postProcessAfterInitialization方法中,对对象进行判断,看他需不需要织入切面逻辑,假如需要,那我就根据这个对象,生成一个代理商对象,而后返回这个代理商对象,那么最终注入容器的,自然就是代理商对象了。
Spring提供了BeanPostProcessor,就是让我们可以对有需要的对象进行“加工解决”啊!
六、认识Spring事务几个重要的接口
Spring事务可以分为两种:
- 编程式事务(通过代码的方式来实现事务)
- 公告式事务(通过配置的方式来实现事务)
编程式事务在Spring实现相对简单少量,而公告式事务由于封装了大量的东西(一般我们使用简单,里头都非常复杂),所以公告式事务实现要难得多。
在编程式事务中有以下几个重要的了接口:
- TransactionDefinition:定义了Spring兼容的事务属性(比方事务隔离级别、事务传播、事务超时、能否只读状态)
- TransactionStatus:代表了事务的具体运行状态(获取事务运行状态的信息,也可以通过该接口间接回滚事务等操作)
- PlatformTransactionManager:事务管理器接口(定义了一组行为,具体实现交由不同的持久化框架来完成—类比JDBC)
PlatformTransactionManager解析
在公告式事务中,除了TransactionStatus和PlatformTransactionManager接口,还有几个重要的接口:
- TransactionProxyFactoryBean:生成代理商对象
- TransactionInterceptor:实现对象的阻拦
- TransactionAttrubute:事务配置的数据
最后
本文主要讲了Spring事务管理少量比较重要的知识点,当然在学习的过程中还看到其余的知识点,假如想要继续学习的同学不妨通过下面给出的参考资料继续阅读。
参考资料:
- 那些年,我们一起追的Spring
- https://zhuanlan.zhihu.com/p/41961670
- 《精通Spring 4.x 企业应用开发实战》
- 《Spring技术内幕》
乐于输出干货的Java技术公众号:Java3y。公众号内有200多篇原创技术文章、海量视频资源、精美脑图,不妨来关注一下!
帅的人都关注了
觉得我的文章写得不错,不妨点一下赞!
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 一文带你认识Spring事务