工作中的许多业务逻辑,可能会用到事务。
事务的概念:
事务,是指作为单个逻辑工作单元执行的一系列操作,结果只有成功和失败两种,要么全部成功(全部提交),要么全部失败(全部回滚),即使成功了一部分,也视为失败,执行全部回滚操作。
事务所具有的四个特性:
原子性(Atomicity):对数据进行操作的时候,要么全部执行,要么全部不执行;
一致性(Consistency):和原子性密切相关,事务执行成功,就使数据库从一个一致性状态改变到另一个一致性状态;
隔离性(Isolation):一个事务的执行,不会被其他事务干扰;
持续性(Durability):事务一旦提交成功,那么数据库里的数据就会永久性地改变;
关于@Transcational注解的理解:
spring支持"编程式事务管理"和"声明式事务管理"两种方式。
而@Transcational注解,就属于使用声明式事务管理,声明式事务管理是建立在AOP之上的,其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务最大的优点是不需要通过编程的方式管理事务,这样就不需要在业务逻辑代码中掺杂事务管理的代码,只需要在配置文件中做相关的事务规则声明(或通过@Transcational注解的方式),便可以将事务规则应用到业务逻辑中。
这种非侵入式的开发方式,是spring所提倡的,也正是这样,声明式事务优于编程式事务。声明式事务唯一的不足是,它最细程度只能达到方法级别,而编程式事务可以作用到代码块级别,弥补的办法是可以将需要进行事务管理的代码块独立为方法等。
使用@Transcational 注解方式:
@Transactional(rollbackFor = Exception.class)
使用场景:@Transcatinal 可以作用于接口、接口方法、类以及类方法上,当作用于类上时,该类的所有public方法都将具有该类型的事务属性,我们也可以在方法级别使用该标注来覆盖类级别的定义。
spring不建议将@Transcational使用在接口或者接口方法上,因为只有在使用基于接口的代理时它才会生效。
另外,@Transcational注解应该只被应用到public方法上,因为只有来自外部的方法调用才会被AOP代理捕获,也就是,类内部方法调用本类内部的其他方法并不会引起事务行为,故只能是public方法,在protected、private或者默认的方法上使用,同时不能使用static的修饰符,若使用则会被忽略,也不会报任何异常。
我们举例使用的@Transactional(rollbackFor = Exception.class),如果标注在一个方法上,表示,在该方法抛出任何异常时,进行事务的回滚动作。
spring不止对捕获数据访问异常才会进行回滚,而是只要捕获到了运行时异常都会进行回滚。
在项目中,@Transactional(rollbackFor=Exception.class),如果类加了这个注解,那么这个类里面的方法抛出异常,就会回滚,数据库里面的数据也会回滚。
在@Transactional注解中如果不配置rollbackFor属性,那么事物只会在遇到RuntimeException的时候才会回滚,加上rollbackFor=Exception.class,可以让事物在遇到非运行时异常时也回滚。
下面展示一种编程式事务管理
public FunctionResult delete(List<String> standardIdList) {
if (standardIdList==null||standardIdList.size()==0){
return new FunctionResult(ErrorCode.SpeExaStandardDeleteListIsNull);
}
TransactionStatus transactionStatus=transactionManager.startTransaction();
int cnt = 0;
for(String standardId:standardIdList) {
try {
cnt = speExaStandardDOMapper.deleteByPrimaryKey(standardId);
}catch (Exception e){
LogHelper.fatal(e.getMessage(),e);
return new FunctionResult(ErrorCode.SpecialExaminationStandardDeleteFail);
}finally{
if (cnt <= 0) {
transactionManager.rollback(transactionStatus);
}else{
transactionManager.commit(transactionStatus);
}
}
}
return new FunctionResult(ErrorCode.Success);
}
关于同一个类中上下层方法的调用,事务注解
https://blog.csdn.net/aya19880214/article/details/50640596?tdsourcetag=s_pctim_aiomsg
事务的相关知识:
https://www.cnblogs.com/zuoxh/p/9724193.html
@Transcational的原理和使用
https://blog.csdn.net/yousite1/article/details/80609992
异常种类区别
https://www.cnblogs.com/clwydjgs/p/9317849.html
https://blog.csdn.net/Mint6/article/details/78363761
https://blog.csdn.net/liaohaojian/article/details/70139151
https://www.cnblogs.com/hjwublog/p/5626465.html
https://www.cnblogs.com/caoyc/p/5632963.html
https://www.cnblogs.com/yepei/p/4716112.html
网友评论