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

问题概述

在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配置文件加密解密工具类分享

    让我们来详细讲解一下如何实现一个Java配置文件加密解密工具类。 1. 需求分析 我们需要一个工具类,能够实现对Java配置文件中的敏感信息进行加密和解密的功能。具体功能如下: 加密配置文件中的敏感信息,保证安全性和保密性; 解密配置文件中的敏感信息,方便在代码中使用; 2. 设计思路 我们的设计思路如下: 读取配置文件,并找到需要加密解密的部分; 对配置文…

    Java 2023年5月31日
    00
  • 简单了解Java编程中抛出异常的方法

    当Java程序中遇到错误或异常时,通常会在程序中使用一些特定的方法来抛出异常并处理异常。本文将详细讲解如何在Java编程中抛出异常的方法。 什么是异常 在了解Java编程中抛出异常的方法之前,我们需要先对“异常”这个概念有一个基本了解。Java编程中抛出的异常代表了一种错误或者问题,例如某个操作出现了意外的输入或输出、某个文件不存在等等。当程序出现异常时,它…

    Java 2023年5月27日
    00
  • WIN7系统JavaEE(java)环境配置教程(一)

    WIN7系统JavaEE(java)环境配置教程(一) 本教程将演示如何在WIN7系统上配置JavaEE(java)开发环境,包括JDK、Eclipse和Tomcat的安装与配置。 第一步:JDK的安装 下载JDK安装包,网址:Oracle官方网站 双击运行安装包,按照提示安装JDK。 配置环境变量,将JDK的bin目录加入系统环境变量中。如下: 属性 -&…

    Java 2023年5月23日
    00
  • 详解Spring Data操作Redis数据库

    详解Spring Data操作Redis数据库 Redis是一种快速、开源的NoSQL数据库,它以键/值(key/value)存储数据,支持多种数据结构,包括字符串、哈希、列表、集合等。在应用程序开发中,连接Redis并进行数据操作是一个常见场景。Spring Data提供了对多种数据存储技术(包括Redis)的抽象和简化,同时还提供了常见的数据操作功能。下…

    Java 2023年5月20日
    00
  • java实现树形菜单对象

    实现树形菜单对象可以采用Java语言和基于树形结构的数据结构,下面是具体的实现攻略: 步骤一:创建树形结构的数据类型 树形菜单对象可以用树形结构数据类型表示,包括各个节点的名称、节点值、父节点、子节点等信息,这个数据结构可以通过类的形式实现: public class TreeNode { private String name; private Objec…

    Java 2023年5月26日
    00
  • 从零开始Mybatis连接数据库的方法

    以下是从零开始Mybatis连接数据库的方法的完整攻略: 步骤1:下载并安装Mybatis Mybatis是一个优秀的持久层框架,我们需要先在官网下载最新版本的Mybatis。下载完成后,按照Mybatis的安装说明进行安装。 步骤2:创建数据库 我们需要在数据库中创建一个表,作为Mybatis连接的对象。这里以MySQL为例,使用以下SQL创建一个user…

    Java 2023年5月19日
    00
  • java实现从方法返回多个值功能示例

    下面是Java实现从方法返回多个值的攻略。 实现方式 Java中可以使用以下几种方式来实现从方法返回多个值的功能: 将多个值封装到一个对象中 使用数组或列表(List) 使用Map 将多个值封装到一个对象中 我们可以定义一个类,将需要返回的多个值封装到它的属性中。例如,假设我们需要返回一个人的姓名、年龄和性别,可以这样定义一个Person类: public …

    Java 2023年5月26日
    00
  • 鼠标焦点离开文本框时验证的js代码

    当用户在网页中填写表单时,我们常常需要验证用户输入的数据是否合法。而当用户在输入框输入完内容后,离开这个输入框,我们需要验证这个输入框中的内容是否符合我们的要求,这时候我们就需要使用JavaScript代码来验证用户的输入。以下是实现鼠标焦点离开文本框时验证的js代码的完整攻略。 1. 绑定事件 我们需要先为输入框绑定一个事件,当输入框失去焦点时触发这个事件…

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