Spring
16 AOP 事务管理
文章目录
- Spring
- 16 AOP 事务管理
- 16.3 Spring 事务属性
- 16.3.1 事务配置
- 16.3.2 转账业务追加日志案例
- 16.3.3 事务传播行为
16.3 Spring 事务属性
16.3.1 事务配置
上面这些属性都可以在@Transactional注解的参数上进行设置。
- readOnly:true只读事务,false读写事务,增删改要设为false,查询设为true。
- timeout:设置超时时间单位秒,在多长时间之内事务没有提交成功就自动回滚,-1表示不设置超时时间。
- rollbackFor:当出现指定异常进行事务回滚
- noRollbackFor:当出现指定异常不进行事务回滚
【思考】
- 出现异常事务会自动回滚,这个是应该的
- noRollbackFor是设定对于指定的异常不回滚,这个好理解
- rollbackFor是指定回滚异常,对于异常事务不应该都回滚么,为什么还要指定?
【!并不是所有的异常都会回滚事务】
【举个例子】
【为什么?】
原因是,Spring的事务只会对Error异常和RuntimeException异常及其子类进行事务回顾,其他的异常类型是不会回滚的,对应IOException不符合上述条件所以不回滚
这个时候就可以使用rollbackFor属性来设置出现IOException异常回滚
- rollbackForClassName等同于rollbackFor,只不过属性为异常的类全名字符
- noRollbackForClassName等同于noRollbackFor,只不过属性为异常的类全名字符串
- isolation设置事务的隔离级别
- DEFAULT :默认隔离级别, 会采用数据库的隔离级别
- READ_UNCOMMITTED : 读未提交
- READ_COMMITTED : 读已提交
- REPEATABLE_READ : 重复读取
- SERIALIZABLE: 串行化
16.3.2 转账业务追加日志案例
【需求分析】
在前面的转案例的基础上添加新的需求,完成转账后记录日志
需求:实现任意两个账户间转账操作,并对每次转账操作在数据库进行留痕
需求微缩:A账户减钱,B账户加钱,数据库记录日志
【如何实现?】
①:基于转账操作案例添加日志模块,实现数据库中记录日志
②:业务层转账操作(transfer),调用减钱、加钱与记录日志功能
【预期效果】
无论转账操作是否成功,均进行转账操作的日志留痕
【环境准备】
① 创建日志表
create table tbl_log(
id int primary key auto_increment,
info varchar(255),
createDate datetime
);
② 添加LogDao接口
package com.dingjiaxiong.dao;
import org.apache.ibatis.annotations.Insert;
/**
* ClassName: LogDao
* date: 2022/9/17 22:52
*
* @author DingJiaxiong
*/
public interface LogDao {
@Insert("insert into tbl_log (info,createDate) values(#{info},now())")
void log(String info);
}
③ 添加LogService接口与实现类
package com.dingjiaxiong.service;
/**
* ClassName: LogService
* date: 2022/9/17 22:53
*
* @author DingJiaxiong
*/
public interface LogService {
void log(String out, String in, Double money);
}
package com.dingjiaxiong.service.impl;
import com.dingjiaxiong.dao.LogDao;
import com.dingjiaxiong.service.LogService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.annotation.Transactional;
/**
* ClassName: LogServiceImpl
* date: 2022/9/17 22:53
*
* @author DingJiaxiong
*/
public class LogServiceImpl implements LogService {
@Autowired
private LogDao logDao;
@Override
@Transactional
public void log(String out, String in, Double money) {
logDao.log("转账操作由 " + out + "到" + in + " ,金额:" + money);
}
}
④ 在转账的业务中添加记录日志
package com.dingjiaxiong.service.impl;
import com.dingjiaxiong.dao.AccountDao;
import com.dingjiaxiong.service.AccountService;
import com.dingjiaxiong.service.LogService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.io.IOException;
/**
* ClassName: AccountServiceImpl
* date: 2022/9/17 22:14
*
* @author DingJiaxiong
*/
@Service
public class AccountServiceImpl implements AccountService {
@Autowired
private AccountDao accountDao;
@Autowired
private LogService logService;
@Override
@Transactional
public void transfer(String out, String in, Double money) throws IOException {
try {
accountDao.outMoney(out,money);
accountDao.inMoney(in,money);
}finally {
logService.log(out,in,money);
}
}
}
⑤ 运行程序
查看日志表
模拟异常
现在的数据情况
【从上面的测试中可以看到】
- 当程序正常运行,tbl_account表中转账成功,tbl_log表中日志记录成功
- 当转账业务之间出现异常(int i =1/0),转账失败,tbl_account成功回滚,但是tbl_log表未添加数据
- 这个结果和咱们想要的不一样,什么原因?该如何解决?
- 失败原因:日志的记录与转账操作隶属同一个事务,同成功同失败
- 想要的最终效果:无论转账操作是否成功,日志必须保留
16.3.3 事务传播行为
不能让日志的事务跟着一起回滚了
【分析】
- log方法、inMoney方法和outMoney方法都属于增删改,分别有事务T1,T2,T3
- transfer因为加了@Transactional注解,也开启了事务T
- 前面讲过Spring事务会把T1,T2,T3都加入到事务T中
- 所以当转账失败后,所有的事务都回滚,导致日志没有记录下来
- 这和咱们的需求不符,这个时候就想能不能让log方法单独是一个事务呢?
要想解决这个问题,就需要用到事务传播行为
【事务传播行为】
事务传播行为:事务协调员对事务管理员所携带事务的处理态度。
【解决上面的问题】
① 修改logService改变事务的传播行为
再次运行程序
查看数据
运行后,就能实现想要的结果,不管转账是否成功,都会记录日志。
【事务传播行为的可选值】
对于开发实际中使用的话,因为默认值需要事务是常态的。根据开发过程选择其他的就可以了,例如案例中需要新事务就需要手工配置。其实入账和出账操作上也有事务,采用的就是默认值。