0. 业务背景
在事务使用时,由于对事务的运行机制不了解,记得有一次在事务使用时,错误的由类的非事务方法调用类自身的事务方法,导致异常回滚失败,事务失效。借此熟悉了下事务的基本流程,同时分析了事务失效的原因。
1. 事务基本流程
事务的基本流程分为2个部分:
- 创建代理对象
data:image/s3,"s3://crabby-images/27789/27789429cdb55ca573ea793c6883f4bf2e4a1181" alt=""
- 在创建完reportServiceImpl这个bean之后,进行bean的后置处理。主要是基于ProxyCreator这个处理器来创建代理对象。
- 通过循环reportServiceImpl这个目标类上面的method,解析缓存method上面的事务属性,如果method上面有事务属性,则代表此切面(增强器)可以使用在目标类上面,因此匹配到bean可以使用的切面(增强器)。
- 基于匹配到切面,来创建代理对象。
- 最后缓存代理对象到单例cache中。
注:这个切面 里面包含了:切点、拦截器(也就是业务方法的增强处理逻辑。后面会说明)。
- 执行事务
data:image/s3,"s3://crabby-images/3e5a4/3e5a4fb7d6cc771b688f234200cd62806b3d5581" alt=""
- context上下文获取代理对象
- 执行业务方法之前,代理对象调用interceptor拦截器的拦截方法
- 拦截里面的业务逻辑:获取业务方法上面的事务属性、创建事务、执行目标方法、异常回滚
2. 原理分析
1. 创建reportServiceImpl这个bean
data:image/s3,"s3://crabby-images/6238a/6238aad8f558ce3346fad8f8ba1b4d72121ee7ba" alt=""
refresh方法入口
data:image/s3,"s3://crabby-images/59811/5981192280eb4e6a3d5e85d4525ab4f1bcfdc3c9" alt=""
data:image/s3,"s3://crabby-images/59707/5970781d1748e090d900a94510b41a8a2c03e339" alt=""
从单例cache中获取名为"reportServiceImpl"单例bean为null,所以调用createBean来创建bean
data:image/s3,"s3://crabby-images/19f06/19f06a84fe864d0d7830d1412e7a89899ceb9607" alt=""
创建bean
data:image/s3,"s3://crabby-images/a7666/a76664e394bff8a8deea1d7d4e9fb1cc31b63817" alt=""
data:image/s3,"s3://crabby-images/2aeb3/2aeb358337baa81d264ba0495e8efa20ad2cad76" alt=""
2. bean的后置处理
data:image/s3,"s3://crabby-images/743d4/743d4f7142bb6548a955b1f89f73e272d34708d4" alt=""
data:image/s3,"s3://crabby-images/e9e97/e9e97f6bae057e0ce072e19ba60380d906e72131" alt=""
处理bean
data:image/s3,"s3://crabby-images/d3157/d3157b00321858cc401d3274f7534e1ecc13343a" alt=""
主要逻辑就是创建代理。
2.1 返回系统缓存的事务属性增强器
这个增强器应该理解为切面。
data:image/s3,"s3://crabby-images/44009/440095f7f478c9b227d37fa54a3e27ab18048e56" alt=""
data:image/s3,"s3://crabby-images/7d38d/7d38dc09726d7c083f95df515f3706edd78f29aa" alt=""
2.2 匹配bean可以使用的切面(增强器)
循环目标类的method
data:image/s3,"s3://crabby-images/973eb/973ebefb8a0d2381e9e45f2c9578eb422eb11dad" alt=""
校验方法为public
data:image/s3,"s3://crabby-images/0be7c/0be7c3d96bb81369f67541dcda901808eddc07fc" alt=""
解析得到事务属性
data:image/s3,"s3://crabby-images/8b1f6/8b1f6f123ffd882faab910f59bb707ab3cabda23" alt=""
data:image/s3,"s3://crabby-images/98a15/98a150aa49e19d7e9dc2ffadf8ca3b37d1b10182" alt=""
返回方法上面的事务属性
data:image/s3,"s3://crabby-images/f6353/f63539f2660648fffabdf3b5cdf8bdde0987018f" alt=""
缓存method->事务属性的映射关系,并返回事务属性。
data:image/s3,"s3://crabby-images/8b6da/8b6da386342662f82090361fcb7b2e72f6f24cf4" alt=""
代码依次返回,直到回到matcher逻辑的这段代码。发现查询目标类的org.example.transaction.service.ReportServiceImpl.generateUseTransactional()方法有事务注解,则返回true
data:image/s3,"s3://crabby-images/2cb29/2cb29cc95568ba631f5dbb43e6bb3a8392804258" alt=""
目标类上有method有事务属性,则保存这个切面并返回。
data:image/s3,"s3://crabby-images/34d11/34d1103e6b57bef778b67edf09d47adb876c4c84" alt=""
代码依次返回,最后返回目标类上面可以使用的切面
data:image/s3,"s3://crabby-images/c5d74/c5d7433bc4a6f496c2903e5dcc4b00f1a66c983e" alt=""
2.3 基于切面创建代理对象
data:image/s3,"s3://crabby-images/3f2ee/3f2ee6f6dd810d60411499835ebc6aa3af5ae330" alt=""
代理工厂指定切面和目标对象
data:image/s3,"s3://crabby-images/a7082/a70822f9e4fe02b62909a51087864ec81b5b357f" alt=""
生成代理对象
data:image/s3,"s3://crabby-images/d7fe0/d7fe0047b27895167cb7f2c8fb801f72c12e4e34" alt=""
data:image/s3,"s3://crabby-images/9c0d3/9c0d3f2fef1087cb5ccf851195004c79c566d880" alt=""
使用enhancer创建代理对象,父类指向目标类。
data:image/s3,"s3://crabby-images/4b1dd/4b1dde679fe99cc5e4f28003b8f7ae258622e2d2" alt=""
创建代理对象,设置回调(即切面里面的拦截器,也就是增强逻辑)
data:image/s3,"s3://crabby-images/7ad47/7ad47265b0238aad06fc98bb295ba8ed2ddc6ea4" alt=""
返回到前面的wrapIfNecessary方法,返回代理对象
data:image/s3,"s3://crabby-images/28038/2803884f1f14c9e8a64ea6295ac31a2541f209e8" alt=""
ProxyCreator处理器处理之后,创建并返回代理对象
data:image/s3,"s3://crabby-images/5bd10/5bd1031cc22cd0e7258436a3eadcef8108457ec8" alt=""
data:image/s3,"s3://crabby-images/93f28/93f28fd3472639da4baedd9c2e52219b5aaddc29" alt=""
流程回到initBean方法,得到代理对象并返回到上层。
data:image/s3,"s3://crabby-images/01d2b/01d2b814d4010c6aeac3de21bc2e5911646a1d00" alt=""
继续返回代理对象
data:image/s3,"s3://crabby-images/6dbfb/6dbfb6c7f0b19233d3a89f5df0a1a5df3e0371c6" alt=""
3. 缓存代理对象到单例cache中
回到getSingleton方法,接收创建的bean(实际上是代理对象),并缓存到单例cache中。
data:image/s3,"s3://crabby-images/52994/52994ed18bff3e63f6587de38e1b7b55da0f330f" alt=""
代理对象 添加到单例的map中
data:image/s3,"s3://crabby-images/fb1ed/fb1ed3c49d61d8e8859a8726137324efd01bbd11" alt=""
4. context上下文获取代理对象
从上面缓存的单例map中获取代理对象
data:image/s3,"s3://crabby-images/bf5c9/bf5c909f61cd0affa17a93064f870d0ba7c34c66" alt=""
data:image/s3,"s3://crabby-images/d528e/d528e024784b8189070d3455428969fec4e741bc" alt=""
继续返回代理对象,直到context得到代理对象
data:image/s3,"s3://crabby-images/7d797/7d797fb79b329aec2bcbd2aa296bf417a178037e" alt=""
data:image/s3,"s3://crabby-images/ebf3f/ebf3fd7830bdfda8a78524564e03615836474325" alt=""
5. 执行事务
回到我们最开始的代码,通过目标对象的代理对象 执行业务方法
data:image/s3,"s3://crabby-images/110ed/110ed22770282681b9275c4179aad2a2ba893240" alt=""
代理对象会调用拦截器的拦截方法,这个拦截方法也就是切面里面业务方法的增强处理逻辑。
data:image/s3,"s3://crabby-images/32c90/32c90d8f54dd8647153f12ddf421c27e2826911d" alt=""
data:image/s3,"s3://crabby-images/d9eae/d9eae4bfad07bfb3ed3629460b36e18b4390d8ff" alt=""
在intercept方法中调用proceed方法
调用事务拦截器
data:image/s3,"s3://crabby-images/70c23/70c23e983ca3beeb09251c4f815de879154b4bb5" alt=""
data:image/s3,"s3://crabby-images/cf72f/cf72f50a3d837a16c44bc96dd0b730f345a62567" alt=""
5.1 获取目标业务方法上面的事务属性
data:image/s3,"s3://crabby-images/a6884/a6884161c9a1c87f736a3f14e02de0c1a50ced32" alt=""
data:image/s3,"s3://crabby-images/0e2df/0e2df37332e98318d14c673c43da3981bf212487" alt=""
5.2 创建事务
data:image/s3,"s3://crabby-images/9e921/9e9211f197702b735089805ebd6f586f7fdcc371" alt=""
data:image/s3,"s3://crabby-images/330d3/330d3e209373cb6599f38cbeb229e75e393434a9" alt=""
【注意】:生成事务的条件。
创建事务对象 ,持有的connection为null
data:image/s3,"s3://crabby-images/87415/874155c82767da618f01d46f298567c832ed2b65" alt=""
开启事务
data:image/s3,"s3://crabby-images/96aa1/96aa1239fd55e8680ef3722eac65db262a9dd3a7" alt=""
获取连接,为事务对象设置连接。
data:image/s3,"s3://crabby-images/a4d7f/a4d7f70fcbdce7df96af24c63ff6fa1fb1ee7a2c" alt=""
关闭自动提交
data:image/s3,"s3://crabby-images/02254/02254b1a63a55181b12dd8833b58ba112be8030c" alt=""
把dataSource数据源->connect映射关系 绑定到当前线程
data:image/s3,"s3://crabby-images/cd140/cd140b8cb0cae3bd7bb560d4c7a7bd3db02a4ac0" alt=""
data:image/s3,"s3://crabby-images/7dc04/7dc04d24fc9ed9a90b36e7136133a56a65b15234" alt=""
设置事务隔离级别,默认。同时激活当前线程的事务同步。
data:image/s3,"s3://crabby-images/59569/5956972c0fca90404f56216e18bd4d677ded4069" alt=""
data:image/s3,"s3://crabby-images/e7f37/e7f37f1ef97c5d2d16d934f9932c9b6e4fa6e53a" alt=""
包装事务信息并绑定到当前线程
data:image/s3,"s3://crabby-images/914c1/914c1ec97e6eb80500521bb4786641862a9aafc8" alt=""
data:image/s3,"s3://crabby-images/af47a/af47a809e7c7122a0ca4c24e0b0300a72f7b55e6" alt=""
5.3 执行目标类中的业务方法
data:image/s3,"s3://crabby-images/9f581/9f58107e590f5d2f1609b11bf848ab1c0c4bd449" alt=""
data:image/s3,"s3://crabby-images/75814/758143ab7536e762ac87e2f36090527ebfcd5f10" alt=""
data:image/s3,"s3://crabby-images/79ac6/79ac68edc18e0e706fd9ce53494ad691a52959c7" alt=""
data:image/s3,"s3://crabby-images/063c8/063c84a11d4f573e7a9e2160103e10227ec7edf6" alt=""
执行到了业务方法中了
data:image/s3,"s3://crabby-images/6c907/6c907e240e2c461ad13c1e592ccf9d0c8bec5f45" alt=""
5.4. 如果异常会执行回滚
data:image/s3,"s3://crabby-images/b9e84/b9e84e980a23874bdba771a4ba8a5a3c0940eed4" alt=""
data:image/s3,"s3://crabby-images/1070f/1070ff2a898fd54ad3f1516c8a4351213f476204" alt=""
检查异常是否是事务属性定义的异常
data:image/s3,"s3://crabby-images/d0040/d0040cc9f7889e67258c2b8f8e7ecb3e469d5f39" alt=""
执行回滚
data:image/s3,"s3://crabby-images/4e3a4/4e3a4570dc68fd189f5868ba08f0a8946da33501" alt=""
连接执行回滚
data:image/s3,"s3://crabby-images/01e7f/01e7fd767d3e745bfed654d4d121ae453c79adef" alt=""
3. 类的非事务方法调用自身的事务方法,事务会失效原因分析
执行非事务注解的业务方法generate()
首先代理对象执行拦截处理,获取可用于generate方法的拦截器
data:image/s3,"s3://crabby-images/ad195/ad1957c5f90739fd6d74e96ca76e90a239080135" alt=""
data:image/s3,"s3://crabby-images/5b904/5b904dfcdaff9b647cace836c0f65429542161a9" alt=""
获取method上是否有事务注解属性
data:image/s3,"s3://crabby-images/0227b/0227b33835b9112006e38caf37e4b803f76d668d" alt=""
很显然没有事务注解,所以返回的拦截器为空
data:image/s3,"s3://crabby-images/8f8f4/8f8f453de6deb316a1c46b486824ae2bcd754555" alt=""
data:image/s3,"s3://crabby-images/54955/54955b7e4bc4a15ff1e7682115e934f4ba10375d" alt=""
data:image/s3,"s3://crabby-images/00eae/00eae6b2db9f58cc1bef8965f8f43e3bd4e534fd" alt=""
所以不会创建事务,更不会执行回滚。
data:image/s3,"s3://crabby-images/ae062/ae06285c8b68df0c2c96de6fe76180ed7701a519" alt=""
data:image/s3,"s3://crabby-images/c6cef/c6cef9f7d835ecf62e93d4e5fd0182a21883f691" alt=""
最终generate方法调用saveReport方法都是以非事务方式运行,所以抛出异常不会回滚。
data:image/s3,"s3://crabby-images/61a9f/61a9f16c21b3e3a3c2ce6fc726d14eba059e1b6e" alt=""
data:image/s3,"s3://crabby-images/68077/68077808831ab8d8186a39b8cb73e2f60bbfc089" alt=""