spring事务处理:调用一个方法前的事务处理过程

2024-06-10 19:38

本文主要是介绍spring事务处理:调用一个方法前的事务处理过程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

实际上,在spring的事务中,只要该类被设置为了事务代理:
 
拦截器都会创建一个TransactionInfo 对象:
 
TransactionInfo txInfo = new TransactionInfo(txAttr, method);
 
而且如果 只要被调用的方法设置了事务属性(txAttr),不管是什么属性都会调用:
 
txInfo.newTransactionStatus(this.transactionManager.getTransaction(txAttr));
 
根据该方法的事务属性(definition )的不同,this.transactionManager.getTransaction(txAttr)的返回值会有所不同(代码见AbstractPlatformTransactionManager),具体为以下几种情况:
1.当前没有事务时(即以下代码中的((HibernateTransactionObject) transaction).hasTransaction()返回false),会返回以下几种:
 
 1    //   Check definition settings for new transaction. 
 2        if   (definition.getTimeout()   <   TransactionDefinition.TIMEOUT_DEFAULT)   {
 3     throw   new  InvalidTimeoutException( " Invalid transaction timeout " , definition.getTimeout());
 4   } 

 5   
 6      //   No existing transaction found -> check propagation behavior to find out how to behave. 
 7        if   (definition.getPropagationBehavior()   ==   TransactionDefinition.PROPAGATION_MANDATORY)   {
 8     throw   new  IllegalTransactionStateException(
 9       " Transaction propagation 'mandatory' but no existing transaction found " );
10   } 

11      else     if   (definition.getPropagationBehavior()   ==   TransactionDefinition.PROPAGATION_REQUIRED   || 
12      definition.getPropagationBehavior()   ==   TransactionDefinition.PROPAGATION_REQUIRES_NEW   || 
13         definition.getPropagationBehavior()   ==   TransactionDefinition.PROPAGATION_NESTED)   {
14      if  (debugEnabled)  {
15     logger.debug( " Creating new transaction with name [ "   +  definition.getName()  +   " ] " );
16    } 

17    doBegin(transaction, definition);
18     boolean  newSynchronization  =  ( this .transactionSynchronization  !=  SYNCHRONIZATION_NEVER);
19     return  newTransactionStatus(definition, transaction,  true , newSynchronization, debugEnabled,  null );
20   } 

21       else     {
22     //  Create "empty" transaction: no actual transaction, but potentially synchronization. 
23      boolean  newSynchronization  =  ( this .transactionSynchronization  ==  SYNCHRONIZATION_ALWAYS);
24     return  newTransactionStatus(definition,  null  false , newSynchronization, debugEnabled,  null );
25   } 

26 
2.当前有事务时
 1    private   TransactionStatus handleExistingTransaction(
 2     TransactionDefinition definition, Object transaction,   boolean   debugEnabled)
 3        throws   TransactionException   {
 4  
 5     if  (definition.getPropagationBehavior()  ==  TransactionDefinition.PROPAGATION_NEVER)  {
 6     throw   new  IllegalTransactionStateException(
 7       " Transaction propagation 'never' but existing transaction found " );
 8   } 

 9  
10     if  (definition.getPropagationBehavior()  ==  TransactionDefinition.PROPAGATION_NOT_SUPPORTED)  {
11      if  (debugEnabled)  {
12     logger.debug( " Suspending current transaction " );
13    } 

14    Object suspendedResources  =  suspend(transaction);
15     boolean  newSynchronization  =  ( this .transactionSynchronization  ==  SYNCHRONIZATION_ALWAYS);
16     return  newTransactionStatus(
17      definition,  null ,  false , newSynchronization, debugEnabled, suspendedResources);
18   } 

19  
20     if  (definition.getPropagationBehavior()  ==  TransactionDefinition.PROPAGATION_REQUIRES_NEW)  {
21      if  (debugEnabled)  {
22     logger.debug( " Suspending current transaction, creating new transaction with name [ "   + 
23       definition.getName()  +   " ] " );
24    } 

25    Object suspendedResources  =  suspend(transaction);
26    doBegin(transaction, definition);
27     boolean  newSynchronization  =  ( this .transactionSynchronization  !=  SYNCHRONIZATION_NEVER);
28     return  newTransactionStatus(
29      definition, transaction,  true , newSynchronization, debugEnabled, suspendedResources);
30   } 

31  
32     if  (definition.getPropagationBehavior()  ==  TransactionDefinition.PROPAGATION_NESTED)  {
33      if  ( ! isNestedTransactionAllowed())  {
34      throw   new  NestedTransactionNotSupportedException(
35        " Transaction manager does not allow nested transactions by default -  "   + 
36        " specify 'nestedTransactionAllowed' property with value 'true' " );
37    } 

38      if  (debugEnabled)  {
39     logger.debug( " Creating nested transaction with name [ "   +  definition.getName()  +   " ] " );
40    } 

41      if  (useSavepointForNestedTransaction())  {
42      //  Create savepoint within existing Spring-managed transaction,
43      //  through the SavepointManager API implemented by TransactionStatus.
44      //  Usually uses JDBC 3.0 savepoints. Never activates Spring synchronization. 
45      DefaultTransactionStatus status  = 
46       newTransactionStatus(definition, transaction,  false ,  false , debugEnabled,  null );
47     status.createAndHoldSavepoint();
48      return  status;
49    } 

50      else   {
51      //  Nested transaction through nested begin and commit/rollback calls.
52      //  Usually only for JTA: Spring synchronization might get activated here
53      //  in case of a pre-existing JTA transaction. 
54      doBegin(transaction, definition);
55      boolean  newSynchronization  =  ( this .transactionSynchronization  !=  SYNCHRONIZATION_NEVER);
56      return  newTransactionStatus(definition, transaction,  true , newSynchronization, debugEnabled,  null );
57    } 

58   } 

59 
最后,txInfo被绑定到当前线程上作为当前事务:
 
txInfo.bindToThread()
 
然后,调用实际的目标类的方法并捕捉异常:
 
  try     {
    //  This is an around advice.
   
 //  Invoke the next interceptor in the chain.
   
 //  This will normally result in a target object being invoked. 
 
   retVal  =  invocation.proceed();
  } 

   
  catch   (Throwable ex)   {
    //  target invocation exception 
 
   doCloseTransactionAfterThrowing(txInfo, ex);
    throw  ex;
  } 

   
  finally     {
   doFinally(txInfo);
  } 

  doCommitTransactionAfterReturning(txInfo);
  
  return   retVal;
 }

另外一点,TransactionInfo的newTransactionStatus调用时如果参数的不是null,TransactionInfo.hasTransaction()方法返回true;
 

这篇关于spring事务处理:调用一个方法前的事务处理过程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1049080

相关文章

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

破茧 JDBC:MyBatis 在 Spring Boot 中的轻量实践指南

《破茧JDBC:MyBatis在SpringBoot中的轻量实践指南》MyBatis是持久层框架,简化JDBC开发,通过接口+XML/注解实现数据访问,动态代理生成实现类,支持增删改查及参数... 目录一、什么是 MyBATis二、 MyBatis 入门2.1、创建项目2.2、配置数据库连接字符串2.3、入

Springboot项目启动失败提示找不到dao类的解决

《Springboot项目启动失败提示找不到dao类的解决》SpringBoot启动失败,因ProductServiceImpl未正确注入ProductDao,原因:Dao未注册为Bean,解决:在启... 目录错误描述原因解决方法总结***************************APPLICA编

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

Python安装Pandas库的两种方法

《Python安装Pandas库的两种方法》本文介绍了三种安装PythonPandas库的方法,通过cmd命令行安装并解决版本冲突,手动下载whl文件安装,更换国内镜像源加速下载,最后建议用pipli... 目录方法一:cmd命令行执行pip install pandas方法二:找到pandas下载库,然后

SpringBoot多环境配置数据读取方式

《SpringBoot多环境配置数据读取方式》SpringBoot通过环境隔离机制,支持properties/yaml/yml多格式配置,结合@Value、Environment和@Configura... 目录一、多环境配置的核心思路二、3种配置文件格式详解2.1 properties格式(传统格式)1.

Apache Ignite 与 Spring Boot 集成详细指南

《ApacheIgnite与SpringBoot集成详细指南》ApacheIgnite官方指南详解如何通过SpringBootStarter扩展实现自动配置,支持厚/轻客户端模式,简化Ign... 目录 一、背景:为什么需要这个集成? 二、两种集成方式(对应两种客户端模型) 三、方式一:自动配置 Thick

Spring WebClient从入门到精通

《SpringWebClient从入门到精通》本文详解SpringWebClient非阻塞响应式特性及优势,涵盖核心API、实战应用与性能优化,对比RestTemplate,为微服务通信提供高效解决... 目录一、WebClient 概述1.1 为什么选择 WebClient?1.2 WebClient 与

Java.lang.InterruptedException被中止异常的原因及解决方案

《Java.lang.InterruptedException被中止异常的原因及解决方案》Java.lang.InterruptedException是线程被中断时抛出的异常,用于协作停止执行,常见于... 目录报错问题报错原因解决方法Java.lang.InterruptedException 是 Jav

深入浅出SpringBoot WebSocket构建实时应用全面指南

《深入浅出SpringBootWebSocket构建实时应用全面指南》WebSocket是一种在单个TCP连接上进行全双工通信的协议,这篇文章主要为大家详细介绍了SpringBoot如何集成WebS... 目录前言为什么需要 WebSocketWebSocket 是什么Spring Boot 如何简化 We