Spring注解@Async和@Transactional失效问题究竟是什么原因,强势解释一波


1. 提前说说

项目中涉及到的代码我都会上传到码云(gitee)或者 github 上,提供给大家下载参考,文中就以最简单的方式说明执行过程。源码的地址在文末哦!

2. 问题场景重现

2.1 场景一:

Spring 的异步执行注解@Async,在调用这个方法的时候发现,不对劲,耗时的逻辑我已经加入到异步去做了,怎么接口请求的响应这么慢,赶紧看日志,懵 X,加了异步注解,却没有异步执行。

2.2 场景二:

在项目中用到@Transactional 注解实现事务是必须滴,如果你还在用 xml 配置,那我只能说……。
但是有时候我们会发现在方法上加了@Transactional 注解却出现灵异事件,在方法内出现异常,数据还是插入到数据库,没有回滚,事务哪里去了,明明是加了的。

3. @Async 注解失效原因分析和解决方案

在看下面的内容之前,对动态代理不是很熟悉的可以看一下我之前的一篇文章(静态代理、动态代理,以及动态代理的调用说明)。
这里添加的注解是通过 Spring AOP 对方法的一种增强,而 Spring AOP 的原理就是动态代理,他的代理有两种,分别是 CGLB 和 JDK 自带的代理,Spring AOP 会根据具体的实现不同,采用不同的代理方式。
动态代理的原理了解了,下面的问题就可以很好的理解。

3.1 异步测试接口

public interface AsyncAopService {
    void addOrder();
    void sendMsg(int result);
}

3.2 异步测试接口实现

@Service
@Slf4j
public class AsyncAopServiceImpl implements AsyncAopService {

    @Autowired
    private OrderDao orderDao;
    @Autowired
    private MsgDao msgDao;

    /**
     * 添加订单后会给用户异步的推送信息
     */
    @Transactional //这里为了让其被代理,加此注解
    public void addOrder() {
        int result = orderDao.insert(OrderModel.builder()
                .amount(10000L)
                .orderId("ORDER_2018042601")
                .phone("15600001212")
                .userId("U_001")
                .build());
        String currentThreadName = Thread.currentThread().getName();
        sendMsg(result);
        System.out.println(currentThreadName + "------>下单结束:mark");
    }

    @Async
    public void sendMsg(int result) {
        try {
            Thread thread = Thread.currentThread();
            thread.sleep(3000);//停留3秒
            String currentThreadName = thread.getName();
            if (result == 1) {
                msgDao.insert(MsgModel.builder().msgContent("下单成功!").receiver("15600001212").build());
                System.out.println(currentThreadName + "------>发送信息成功");
            } else {
                msgDao.insert(MsgModel.builder().msgContent("下单失败!").receiver("15600001212").build());
                System.out.println(currentThreadName + "------>发送信息失败");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

3.3 测试类

@Test
public void AsyncTest() throws InterruptedException {
	System.out.println("=======async test start=======");
	asyncAopService.addOrder();
	System.out.println("=======async test end=======");
	/**
	 * 在这里让线程睡5秒的原因是为了能够看到异步执行的结果日志
	 * 小知识点:在Junit测试中,如果主线程执行结束,整个测试过程也结束了,存在的异步逻辑如果没有执行完就不会执行啦!
	 * 测试方式:把这行代码去掉,执行测试,t_order_info表中会插入数据,但是t_msg_info表中无数据插入。
	 */
	Thread.sleep(5000);
}

3.4 测试类上的注解

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)

Spring Boot 启动类的注解:

@SpringBootApplication
@MapperScan("com.minuor.aop.dao")
@EnableAspectJAutoProxy
@EnableAsync //开启异步功能

测试结果

=======async test start=======
main------>发送信息成功
main------>下单结束:mark
=======async test end=======

运行的过程很慢,两行日志的线程名称相同,并且 mark 日志是在发送信息成功后再输出的,回到代码可以知道发送信息逻辑是异步的执行,为什么会和下单过程的线程名称相同,并且异步执行发送消息是延迟的,为何日志还在 mark 日志前。种种迹象表明,这不是一个异步执行,而是顺序执行。但是这里是加了异步的注解的,实际没有生效。
addOrder()方法里面直接调用sendMsg(……)方法,这里还隐含一个关键字,那就是 this,实际上这里调用是这样的:this.sendMsg(),this 是当前对象。而addOrder()是被代理的,在代理对象中执行结束增强后,通过invoke,用实际AsyncAopServiceImpl对象来调用addOrder()方法执行业务逻辑。在业务逻辑内又调用了sendMsg(……)方法,调用的对象是当前对象,当前对象是AsyncAopServiceImpl,问题就出在这里,因为要想用异步执行sendMsg(……),必须用代理对象执行,因为代理对象要做异步相关的增强,但是此时却直接用AsyncAopServiceImpl对象调用,绕过了代理对象增强的部分,也就是说代理增强部分失效,@Async注解失效。原来想异步执行的逻辑,变成了顺序执行。

3.5 解决方案

没有用代理对象执行sendMsg(……),被AsyncAopServiceImpl对象抢占了先机。那么解决就是要让代理对象来执行sendMsg(……)

3.5.1 在调用 sendMsg(……)之前添加下面的代码

// 获取代理对象,通过AOP上下文
AsyncAopService service = (AsyncAopService) AopContext.currentProxy();
service.sendMsg(result); //通过代理对象调用sendMsg,做异步增强

这里还不算完,如果就这样运行,那肯定会报错。

3.5.2 在@EnableAspectJAutoProxy 添加属性值。

@EnableAspectJAutoProxy(exposeProxy = true)

3.5.3 运行结果

=======async test start=======
main------>下单结束:mark
=======async test end=======
SimpleAsyncTaskExecutor-1------>发送信息成功

结果也是想要的结果,下单结束,整个测试结束,在测试结束后等待 5 秒,等待异步日志打印。主线程和异步线程是不同的两个。
如果对代理对象和当前对象有点懵的话,可以加上下面的两行代码:

System.out.println("------>代理对象:"+service.getClass());
System.out.println("------>当前对象:"+this.getClass());

得到的结果:

------>代理对象:class com.minuor.aop.impl.AsyncAopServiceImpl$$EnhancerBySpringCGLIB$$9de92f4b //可以看出来是CGLB动态代理
------>当前对象:class com.minuor.aop.impl.AsyncAopServiceImpl

4. @Transactional 注解失效的原因分析

这个原因和上面的是相同的,代理被绕过,直接当前对象执行应该被增强的方法,导致方法没有被增强成功。但是可以说一下两个情况。

4.1 情况一:在非代理增强方法中调用加了@Transactional 增强的方法

这个过程容易理解,不解释。

4.1.1 业务代码

@Service
public class TransactionalAopServiceImpl implements TransactionalAopService {

    @Autowired
    private OrderDao orderDao;
    @Autowired
    private UserDao userDao;

    public void addOrder() {
        orderDao.insert(OrderModel.builder()
                .userId("YK_002") //游客编号
                .phone("13522203330")
                .orderId("ORDER_2018042602")
                .amount(10000L)
                .build());
        //默开用户
        System.out.println("--->"+this.getClass());
        addUser("13522203330");
    }

    @Transactional
    public void addUser(String phone) {
        userDao.insert(UserModel.builder().userName("zhangsan").userPhone(phone).build());
        throw new RuntimeException();
    }
}

运行结果是 order 订单信息添加成功,同时 user 用户信息也添加成功,数据库都有数据,没有回滚。按照表面理解应该是 order 添加成功,user 添加失败,因为 addUser 上加了事务,会回滚。原理参照@Async 失效的原理解释。

4.2 情况二:addOrder 和 addUser 方法上都添加@Transactional

两个方法都添加了@Transactional方法,到底是如何回滚的呢,这里涉及很多情况。

  • 都加了@Transactional 注解,但是在调用方法的时候不是用的代理对象,这个时候会根据调用者的注解规则进行回滚;
  • 都加了@Transactional 注解,且调用方法的时候使用的代理对象,这个时候就涉及 spring 的事务传播机制,需要根据不同情况做不同的分析。

5. 源代码

具体的代码是在com.minuor.aop包,测试类在 test 下com.minuor.MinuorAnnotationService
gitHub:https://github.com/itcrud/proxy


文章作者: 程序猿洞晓
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 程序猿洞晓 !
评论
 上一篇
JDK7、8对ConcurrentHashMap的实现和总结 JDK7、8对ConcurrentHashMap的实现和总结
ConcurrentHashMap是一个经常被使用的数据结构,相比于Hashtable以及Collections.synchronizedMap(),ConcurrentHashMap在线程安全的基础上提供了更好的写并发能力,但同时降低了对读一致性的要求(这点好像CAP理论啊 O(∩_∩)O)。ConcurrentHashMap的……
2018-06-20
下一篇 
Spring AOP 动态代理 CGLIB、JDK 的基本原理 Spring AOP 动态代理 CGLIB、JDK 的基本原理
AOP(Aspect Orient Programming),我们一般称为面向方面(切面)编程,作为面向对象的一种补充,用于处理系统中分布于各个模块的横切关注点,比如事务管理、日志、缓存等等。AOP 实现的关键在于 AOP 框架自动创建的 AOP 代理,AOP 代理主要分为静态代理和动态代理,静态代理的代表为 AspectJ;而动态代理则以 Spring AOP 为代表。本文会分别对 AspectJ 和……
2018-06-15
  目录