spring-transaction源码分析(3)Transactional事务失效原因

yizhihongxing

问题概述

在Transactional方法中使用this方式调用另一个Transactional方法时,拦截器无法拦截到被调用方法,严重时会使事务失效。

类似以下代码:

@Transactional
public void insertBlogList(List<Blog> blogList) {
  for (Blog blog : blogList) {
    this.blogMapper.insertBlog(blog);
  }
  try {
    TimeUnit.SECONDS.sleep(15);
  } catch (InterruptedException e) {
    e.printStackTrace();
  }
}

@Transactional
public void deleteBlogByCondition(BlogSearchParameter parameter) {
  List<Blog> blogs = this.blogMapper.selectBlogByParameter(parameter);
  for (Blog blog : blogs) {
    this.blogMapper.deleteBlog(blog.getId());
  }
  // 抛出一个RuntimeException
  throw new RuntimeException("deleteBlogByCondition抛出一个异常");
}

@Transactional
public void insertAndDeleteBlogList2(List<Blog> blogList, BlogSearchParameter parameter) {

  // 插入数据
  this.insertBlogList(blogList);

  // 删除数据
  try {
    this.deleteBlogByCondition(parameter);
  } catch (Exception e) {
    System.err.printf("Err:%s%n", e.getMessage());
  }

  System.out.println("继续插入数据");

  // 继续插入数据
  this.insertBlogList(blogList);
}

正常情况下,执行到"继续插入数据"时会抛出一个"rollback only"的异常,然后事务回滚。

而现在的现象是:

  • 三个操作都不会开启事务,出现异常也不会回滚
  • "删除数据"操作会把符合条件的数据都删除掉
  • "继续插入数据"操作会再插入数据

原因分析

在EnableTransactionManagement注解mode属性的文档中:

The default is AdviceMode.PROXY. Please note that proxy mode allows for interception of calls through the proxy only.
Local calls within the same class cannot get intercepted that way; an Transactional annotation on such a method within 
a local call will be ignored since Spring's interceptor does not even kick in for such a runtime scenario.
For a more advanced mode of interception, consider switching this to AdviceMode.ASPECTJ.

大概意思是:mode属性的默认值是AdviceMode.PROXY,这种方式仅允许通过代理对来调用事务方法,同一个类的本地调用无法被事务切面拦截。如果要解决这个问题,可以使用AdviceMode.ASPECTJ模式。

其实这个问题的根本原因与spring-tx无关,而是spring-aop的实现方式造成的。

从spring-aop拦截器分析问题原因

在DynamicAdvisedInterceptor和JdkDynamicAopProxy中有一段类似的代码:

spring-transaction源码分析(3)Transactional事务失效原因

spring-transaction源码分析(3)Transactional事务失效原因

其中target就是原始的业务层Bean对象。

在后续创建ReflectiveMethodInvocation/CglibMethodInvocation时又将此target传递了进去:

// JDK
MethodInvocation invocation =
		new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);
retVal = invocation.proceed();

// Cglib
retVal = new CglibMethodInvocation(proxy, target, method, args, targetClass, chain, methodProxy).proceed();

proceed方法中在拦截器链最后会调用目标方法:

public Object proceed() throws Throwable {
	// We start with an index of -1 and increment early.
	if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
		return invokeJoinpoint();
	}

	// 略
}

protected Object invokeJoinpoint() throws Throwable {
	// 反射调用目标方法
	// 这个target就是原始Bean对象
	return AopUtils.invokeJoinpointUsingReflection(this.target, this.method, this.arguments);
}

所以如果在目标方法中使用this方法调用另一个需要被拦截的方法,将不会执行拦截逻辑。

原文链接:https://www.cnblogs.com/xugf/p/17390167.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:spring-transaction源码分析(3)Transactional事务失效原因 - Python技术站

(0)
上一篇 2023年5月11日
下一篇 2023年5月11日

相关文章

  • java 重定义数组的实现方法(与VB的ReDim相像)

    问题:详细讲解“java 重定义数组的实现方法(与VB的ReDim相像)”的完整攻略,过程中至少包含两条示例说明。 回答: 在Java中,数组的长度一旦确定后是不可变的,但有些情况下可能需要动态地改变数组的长度,这就需要对数组进行重新定义。本文将介绍Java中重定义数组的实现方法(与VB的ReDim相似)。 方法一:使用Arrays.copyOf方法 Arr…

    Java 2023年5月26日
    00
  • springboot jackson配置教程

    下面是SpringBoot Jackson配置教程的完整攻略,包括Jackson的基本功能、注解、配置、示例等详细步骤。 一、什么是Jackson Jackson是Java编程语言中用于将Java对象序列化为JSON(JavaScript对象表示)和反序列化从JSON返回Java对象的库。它是一种流行的JSON库,可以快速轻松地将Java对象转换为JSON格…

    Java 2023年5月26日
    00
  • Java反射机制及Method.invoke详解

    Java反射机制及Method.invoke详解 什么是Java反射机制? Java反射机制是指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意方法和属性。这种动态获取信息以及动态调用对象的功能称为Java反射机制。 Java反射机制的核心是java.lang.reflect包,该包下的Class类、Met…

    Java 2023年5月26日
    00
  • struts2标签总结_动力节点Java学院整理

    Struts2标签总结攻略 Struts2是一个MVC框架,它提供了很多标签来简化视图层的开发。本文将对Struts2的标签进行总结,提供代码示例。 简介 Struts2标签库可分为以下几种类型: 表单标签:form, textfield, password, textarea, checkbox, radio, select,option等。 表单验证标签…

    Java 2023年5月20日
    00
  • javaweb购物车案列学习开发

    首先我们来讲一下”javaweb购物车案例学习开发”的完整攻略。 知识储备 在学习JavaWeb购物车之前,需要具备一些Java基础知识,比如面向对象编程、基本的Java语法、JSP基础语法、Servlet基础知识、JDBC数据库操作等。如果没有接触过这些知识,请先学习相关基础课程。 构思 在开始开发之前,需要有一个完整的购物车设计。购物车的基本结构包括商品…

    Java 2023年6月2日
    00
  • MyBatis中多条件查询商品的三种方法及区别

    MyBatis中多条件查询商品的三种方法及区别 在开发中,往往需要根据多个条件来查询数据。MyBatis提供了多种方法来实现多条件查询,本文将介绍三种方法并分析它们之间的差异。 方法一:使用<if>标签 使用<if>标签的方式适用于查询条件较少的情况。我们需要在SQL语句中使用<if>标签来判断条件是否成立,如果成立则拼接…

    Java 2023年5月20日
    00
  • 浅谈Java响应式系统

    浅谈Java响应式系统 什么是Java响应式系统 Java响应式系统是一种基于反应式编程(Reactive Programming,简称RP)思想的编程模式,它使用异步流来构建可靠性高,性能较好的应用程序。在Java响应式系统中,数据流可以被看作是一系列的事件或者消息,应用程序可以通过订阅这些事件或者消息来进行处理。Java响应式系统常常被用于处理大规模数据…

    Java 2023年5月20日
    00
  • Java的Struts框架报错“ControllerResourcesNotFoundException”的原因与解决办法

    当使用Java的Struts框架时,可能会遇到“ControllerResourcesNotFoundException”错误。这个错误通常由以下原因之一起: 配置错误:如果配置文件中没有正确配置,则可能会出现此错误。在这种情况下,需要检查文件以解决此问题。 控制器错误:如果控制器不正确,则可能会出现此错误。在这种情况下,需要检查控制器以解决此问题。 以下是…

    Java 2023年5月5日
    00
合作推广
合作推广
分享本页
返回顶部