这是我参与11月更文挑战的第10天,活动详情查看:2021最后一次更文挑战
前言
事务是一个需要同时保证原子性、隔离性、一致性和持久性的一个或多个数据库操作
本文会说明,springBoot中两种事务的实现方式,编程式事务配置和声明式事务配置,当然在此之前会说一些基础的东西:事务的四大特征,事务的隔离级别,事务的传播行为
事务的四大特征(ACID)
- 原子性(Atomicity):事务中的全部操作在数据库中是不可分割的,要么全部执行,要么均不执行.
- 一致性(Consistency):几个并行执行的事务,其执行结果必须与按某一顺序串行执行的结果相一致.
- 隔离性(Isolation):事务的执行不受其他事务的干扰,事务执行的中间结果对其他事务必须是透明的.
- 持久性(Durability):对于任意已提交事务,系统必须保证该事务对数据库的改变不被丢失.
开启事务
@EnableTransactionManagement
复制代码
SpringBoot声明式事务
声明式事务@Transactional
可以使用在类上,也可以使用在public
方法上. 如果是使用在类上,则是对所有的public
方法都开启事务,如果类和方法上都有则方法上的事务生效
可以在类上使用
@Transactional(rollbackFor=Exception.class)
public class TransactionServiceImpl implements TransactionService {
}
复制代码
更多的是在方法上使用
@Override
@Transactional(rollbackFor=Exception.class)
public void t1(Student one) {
}
复制代码
@Transactional
的参数
属性 | 作用 |
---|---|
value | 当配置了多个事务管理器时,可以使用该属性指定选择哪个事务管理器. |
transactionManager | 当配置了多个事务管理器时,可以使用该属性指定选择哪个事务管理器. |
propagation | 事务的传播行为,默认值为 Propagation.REQUIRED |
isolation | 事务的隔离级别,默认值为 Isolation.DEFAULT |
timeout | 事务的超时时间,默认值为-1.如果超过该时间限制但事务还没有完成,则自动回滚事务. |
readOnly | 指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置read-only 为 true. |
rollbackFor | 用于指定能够触发事务回滚的异常类型,可以指定多个异常类型. |
rollbackForClassName | 用于指定能够触发事务回滚的异常类型,可以指定多个异常类型. |
noRollbackFor | 抛出指定的异常类型,不回滚事务,也可以指定多个异常类型. |
noRollbackForClassName | 抛出指定的异常类型,不回滚事务,也可以指定多个异常类型. |
value
和transactionManager
相同 rollbackFor
和rollbackForClassName
相同 noRollbackFor
和noRollbackForClassName
相同
事务的隔离级别(isolation)
事务隔离级别 | 脏读 | 不可重复读 | 幻读 |
---|---|---|---|
读未提交(read-uncommitted) | 是 | 是 | 是 |
不可重复读(read-committed) | 否 | 是 | 是 |
可重复读(repeatable-read) | 否 | 否 | 是 |
串行化(serializable) | 否 | 否 | 否 |
事务的传播行为(propagation)
传播行为 | 解释 |
---|---|
REQUIRED | 如果当前存在事务,则加入该事务,如果当前不存在事务,则创建一个新的事务. |
SUPPORTS | 如果当前存在事务,则加入该事务;如果当前不存在事务,则以非事务的方式继续运行. |
MANDATORY | 如果当前存在事务,则加入该事务;如果当前不存在事务,则抛出异常. |
REQUIRES_NEW | 重新创建一个新的事务,如果当前存在事务,暂停当前的事务. |
NOT_SUPPORTED | 以非事务的方式运行,如果当前存在事务,暂停当前的事务. |
NEVER | 以非事务的方式运行,如果当前存在事务,则抛出异常. |
NESTED | 和REQUIRED效果一样. |
SpringBoot编程式事务
此种方式基于AOP
功能,所以需要添加
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
复制代码
写配置类
@Aspect
@Configuration
public class MyTransactionConfig {
/**
* 配置方法过期时间,默认-1,永不超时
*/
private final static int TX_METHOD_TIME_OUT = 10;
/**
* 全局事务位置配置 在哪些地方需要进行事务处理
* 配置切入点表达式
*/
private static final String POITCUT_EXPRESSION = "execution(* zdc.enterprise.service.impl.*.*(..))";
@Autowired
private PlatformTransactionManager platformTransactionManager;
@Bean
public TransactionInterceptor txadvice() {
/*只读事物、不做更新删除等*/
/*事务管理规则*/
RuleBasedTransactionAttribute readOnlyRule = new RuleBasedTransactionAttribute();
/*设置当前事务是否为只读事务,true为只读*/
readOnlyRule.setReadOnly(true);
/* transactiondefinition 定义事务的隔离级别;
*如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。*/
readOnlyRule.setPropagationBehavior(TransactionDefinition.PROPAGATION_SUPPORTS);
/*增删改事务规则*/
RuleBasedTransactionAttribute requireRule = new RuleBasedTransactionAttribute();
/*抛出异常后执行切点回滚 建议自定义异常*/
requireRule.setRollbackRules(Collections.singletonList(new RollbackRuleAttribute(Exception.class)));
/*PROPAGATION_REQUIRED:事务隔离性为1,若当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。这是默认值。 */
requireRule.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
/*设置事务失效时间,超过10秒*/
requireRule.setTimeout(TX_METHOD_TIME_OUT);
/** 配置事务管理规则
nameMap声明具备需要管理事务的方法名.
这里使用addTransactionalMethod 使用setNameMap
*/
Map<String, TransactionAttribute> nameMap = new HashMap<>();
nameMap.put("add*", requireRule);
nameMap.put("save*", requireRule);
nameMap.put("insert*", requireRule);
nameMap.put("update*", requireRule);
nameMap.put("delete*", requireRule);
nameMap.put("remove*", requireRule);
/*进行批量操作时*/
nameMap.put("batch*", requireRule);
nameMap.put("get*", readOnlyRule);
nameMap.put("query*", readOnlyRule);
nameMap.put("find*", readOnlyRule);
nameMap.put("select*", readOnlyRule);
nameMap.put("count*", readOnlyRule);
NameMatchTransactionAttributeSource source = new NameMatchTransactionAttributeSource();
source.setNameMap(nameMap);
TransactionInterceptor transactionInterceptor = new TransactionInterceptor(platformTransactionManager, source);
return transactionInterceptor;
}
/**
* 设置切面=切点pointcut+通知TxAdvice
* @return
*/
@Bean
public Advisor txAdviceAdvisor() {
AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();
pointcut.setExpression(POITCUT_EXPRESSION);
return new DefaultPointcutAdvisor(pointcut, txadvice());
}
}
复制代码
有了这个切面配置类,就不要用在类或者每个方法上使用@Transactional
了,当然方法名前缀要能和设置的匹配上. RuleBasedTransactionAttribute
的参数大致和@Transactional
的参数相同,里面有详细的注释,就不过多解释了
作者:ZOUZDC
链接:https://juejin.cn/post/7028963866063306760
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
复制代码