service层@Transactional事务控制使用以及注意

 

@Service
@Transactional
public class CeshiService {


    @Autowired
    LoginFeignClient loginFeignClient;

    @Transactional(rollbackFor = Exception.class)
    public void test(){
        User user=null;
        //try {
            log.info("ceshi transactional is 1");
            user=loginFeignClient.test1().getData();
            throw  new RuntimeException();
            //log.info("ceshi transactional is 2");
           // loginFeignClient.test();
       // }catch (Exception e){
        //    throw new Exception(e.getLocalizedMessage());
       // }

    }
}


上述就是一个事务的service,当获取到throw的exception时执行rollback;

 

1.service中是通过feign调用远程服务时,上述写法是无法回滚的,只有repository在本地代码处理才可以,因为db提供了回滚处理。

2.如果是feign远程调用,又想使用事务,则需要进行逆服务处理,需要获取异常时,手动调用逆服务处理

3.非事务声明方法调用事务声明方法,则事务失效。使用了@Transactional的方法,对同一个类里面的方法调用, @Transactional无效。比如有一个类Test,它的一个方法A,A再调用Test本类的方法B(不管B是否public还是private),但A没有声明注解事务,而B有。则外部调用A之后,B的事务是不会起作用的。(经常在这里出错)

针对这种情况,进行了实例测试:

@Service
@Slf4j
public class TransTestService {

    @Autowired
    QueueAppointRecordMapper  queueAppointRecordMapper ;

    public void noTrans(){
        QueueAppointRecordVo recordVo = new QueueAppointRecordVo();
        recordVo.setId(6549928413648389141L);
        List list1 = queueAppointRecordMapper.getAppointRecordById(recordVo);
        log.info("record status1:{}", list1.get(0).getStatus());
        QueueAppointRecordVo vo = new QueueAppointRecordVo() ;
        vo.setId(6549928413648389141L);
        vo.setStatus("R07");
        queueAppointRecordMapper.updateStatus(vo) ;
        log.info("noTrans,status:R07");
        trans();
    }

    @Transactional
    public void trans(){
        QueueAppointRecordVo vo = new QueueAppointRecordVo() ;
        vo.setId(6549928413648389141L);
        vo.setStatus("R09");
        queueAppointRecordMapper.updateStatus(vo) ;
        log.info("trans,status:R09");
        throw new RuntimeException() ;
    }
}

没有@Transactional标签的方法notrans调用有@Transactional的方法trans方法,

这条记录原始状态R01,经过程序后依次变更为R07  R09,虽然trans事务方法遇到了runtimeException,最后查数据库的状态也是R09。也就说明trans方法的事务是没有起作用的。

service层@Transactional事务控制使用以及注意_第1张图片

service层@Transactional事务控制使用以及注意_第2张图片

service层@Transactional事务控制使用以及注意_第3张图片

4.一个service中声明事务方法,调用了另外一个service中的声明事务方法,则被调用的方法事务也还起作用,事务不太建议放到2个见service中。下面的例子,下面的事务是起作用的,跟上面第3条写的注意区分:

同一工程内,controller中调用第一个service声明了Transactional,service中再调用其他service中非声明的事务,如果遇到RunTimeException则还是事务回滚的,但是尽量使用事务的方法,都加上Transactional声明

 

 

你可能感兴趣的:(java,spring)