Spring中解决事务以及异步注解失效

一、重现@Transaction失效的场景

        有如下业务场景,新增订单后,自动发送短信,下面的代码在同一个类中:

@Transaction
public void addOrder(OrderInfo  orderInfo){ 
        orderMapper.insert(orderInfo);
        try{
                sendMesg(orderInfo );
        }cach(Exception e){
                e.printStrace();
        }
}
@Transaction
public void sendMesg( OrderInfo  orderInfo ){
        mesgMapper.insert(orderInfo );
        throws new RuntimeException("发送短信出现异常!");
}

        上面的伪代码模拟新增订单后,自动发送短信的业务场景。两个操作被标识为事务,为了不影响发送短信出现异常影响订单的插入,在调用发送短信的方法时,通过 try....cach...捕获其异常并处理,不影响订单表的插入。

        因为sendMesg标识为事务,其抛出异常后,事务按正常逻辑来说,事务会进行回滚,即短信表中不会插入记录。然而事与愿违,出现的结果是短信表也插入了记录。

二、重现异步注解失效的场景

1、异步注解@Async介绍

        基于@Async标注的方法,称之为异步方法,这些方法在执行的时候,spring将会为其开辟独立的线程执行,调用者无需等待它的完成,即可继续其他的操作。

2、如何使用@Async

        增加 aspectj 相关的依赖;

        修改 spring配置文件,在配置文件中增加如下配置:

        <task:annotation-driven executor = "annotationExecutor"  />

        <task:executor id="annotationExecutor" pool-size="20" />

        在方法上加上@Async注解。

3、业务场景介绍

预定旅游套餐业务场景

@Transaction 
public void planTourismPackages(Order orderInfo){
        dealPassengerTicketBusiness(orderInfo);
        dealHotelBusiness(orderInfo);
        sendMesg(orderInfo);
        sendEmial(orderInfo);
 }
@Async
public void sendMesg( OrderInfo  orderInfo ){
        mesgMapper.insert( orderInfo  );         
        throws new RuntimeException("发送短信出现异常!"); 
}

        预定旅游套餐时,首先处理机票预定业务,然后处理酒店预定业务,最后发送短信和邮件。发送短信和邮件属于辅助业务,可以让其异步执行,所以在方法上加了@Async注解,让其异步执行。为了演示效果,我将planTourismPackages方法标识为事务处理,故意让sendMesg方法抛出异常。按我们预想,因为sendMesg方法异步执行,其抛出异常应该不会影响planTourismPackages事务的提交,机票预定表和酒店预定表应该会插入记录。但是,事与愿违,sendMesg抛出异常导致planTourismPackages事务会滚了,说明@Async并未生效。

三、原因分析

        为什么会出现上面的事务和异步注解失效呢?

        spring声明式事务和异步注解的实现都是基于spring aop,即对标识的方法进行增强。spring aop的底层实现原理是 jdk 动态代理。因为事务注解方法,异步方法的调用的方法在同一个类中,所以发送短信方法是在调用方法的代理对象中执行的,没有对发送短信方法进行增强。如下图:

orderMapper.insert( orderInfo );                                                     
try{ 
         this.sendMesg( orderInfo  ); 
 }cach(Exception e){ 
        e.printStrace();   
 } 
public void planTourismPackages(Order orderInfo){                                                          
        dealHotelBusiness( orderInfo ); 
        dealPassengerTicketBusiness( orderInfo );  
        this.sendMesg( orderInfo );
        this.sendEmial( orderInfo );  
}

四、解决方案

        要解决上面的问题,必须要实现发送短信方法的增强,使其能够成为事务或者异步方法,即让代理生效。spring的解决方案如下:

        1、在spring配置文件xml新增如下语句:

        先开启cglib代理,开启 exposeProxy=true,暴露代理对象

        <aop:aspectj-autoproxy expose-proxy="true"/>

        2、使用AopContext 获取当前对象的动态代理。

        修改配置文件后,代码修改,用获取到的动态代理去执行发送短信方法,如下:

TourServer currentProxy = (TourServer )AopContext.currentProxy();
        currentProxy.sendMesg(orderInfo);
        currentProxy.sendEmail(orderInfo);