事务以及@Transcational注解

2018-12-18  本文已影响0人  我就要取名叫夏末

工作中的许多业务逻辑,可能会用到事务。
事务的概念:
事务,是指作为单个逻辑工作单元执行的一系列操作,结果只有成功和失败两种,要么全部成功(全部提交),要么全部失败(全部回滚),即使成功了一部分,也视为失败,执行全部回滚操作。

事务所具有的四个特性:

原子性(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

上一篇下一篇

猜你喜欢

热点阅读