SpringAop学习笔记(二)——Spring事务回滚的原理

简介: SpringAop学习笔记(二)——Spring事务回滚的原理

正文


一、代码


package com.xiaojie.annotation;
import java.lang.annotation.*;
/**
 * 自定义事务注解
 */
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface MyTransactional {
}


package com.xiaojie.util;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Scope;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.stereotype.Component;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.interceptor.DefaultTransactionAttribute;
/**
 * @author Administrator
 *编程事务类,手动开始,手动提交,手动回滚
 */
@Component
@Scope("prototype")
public class TransactionUtils {
  private TransactionStatus status;
  @Autowired
  private DataSourceTransactionManager dataSourceTransactionManager;
    //开始事务
    public TransactionStatus begin() {
      TransactionStatus transaction = dataSourceTransactionManager.getTransaction(new DefaultTransactionAttribute());
      return transaction;
    }
    //提交事务
    public void commit(TransactionStatus transaction) {
      dataSourceTransactionManager.commit(transaction);
    }
    //回滚事务
    public void rollback() {
      System.out.println("事务回滚。。。。。。");
      dataSourceTransactionManager.rollback(status);
    }
}


package com.xiaojie.aop;
import com.xiaojie.annotation.MyTransactional;
import com.xiaojie.util.TransactionUtils;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import org.springframework.transaction.TransactionStatus;
import java.lang.reflect.Method;
/**
 * 自定义事务切面,原理
 * 通过环绕通知,当执行目标方法没有异常时提交事务
 * 当执行目标方法发生异常时,在异常通知中回滚事务,实现事务提交与回滚
 */
@Aspect
@Component
public class TransactionalAop {
    @Autowired
    private TransactionUtils transactionUtils;
    /**
     * 定义切入点
     */
    @Pointcut("execution(* com.xiaojie.service.*.*.*(..))")
    public void transactionalPoint(){}
    @Around("transactionalPoint()")
    public Object around(ProceedingJoinPoint point) throws Throwable {
        //获取方法名称
        String methodName = point.getSignature().getName();
        // 获取目标对象
        Class<? extends Object> targetClass = point.getTarget().getClass();
        // 获取目标对象类型
        Class[] par = ((MethodSignature) point.getSignature()).getParameterTypes();
        // 获取目标对象方法
        Method objMethod = targetClass.getDeclaredMethod(methodName, par);
        //获取目标方法上的注解
        MyTransactional declaredAnnotation = objMethod.getDeclaredAnnotation(MyTransactional.class);
        //如果方法上有注解则开始事务
        if (null==declaredAnnotation){
            //如果没有注解,则执行目标方法
            return point.proceed();
        }
        //有注解,开启事务
        System.out.println("开启事务。。。。。。。");
        TransactionStatus status = transactionUtils.begin();
        //调用目标方法
        Object proceed = point.proceed();
        if (status!=null){
            //提交事务
            transactionUtils.commit(status);
        }
        return proceed;
    }
    /**
     * 目标方法发生异常,则执行回滚事务
     */
    @AfterThrowing("transactionalPoint()")
    public void afterthrow() {
        //如果发生异常则回滚事务
        transactionUtils.rollback();
    }
}


完整代码请参考https://gitee.com/whisperofjune/spring-transactional.git


二、Spring 事务失效之谜


spring的事务注解@Transactional只能放在public修饰的方法上才起作用,如果放在其他非public(private,protected)方法上,虽然不报错,但是事务不起作用


如使用mysql且引擎是MyISAM,则事务会不起作用,原因是MyISAM不支持事务,可以改成InnoDB引擎


在业务代码中如果抛出RuntimeException异常,事务回滚;但是抛出Exception,事务不回滚;需要在注解上配置


@Transactional(rollbackFor={RuntimeException.class,Exception.class})


如果在加有事务的方法内,使用了try...catch..语句块对异常进行了捕获,而catch语句块没有throw  new RuntimeExecption异常,事务也不会回滚。因为aop捕获不到异常,不能进行异常通知从而不能执行事务回滚的方法。


在类A里面有方法a 和方法b, 然后方法b上面用 @Transactional加了方法级别的事务,在方法a里面 调用了方法b, 方法b里面的事务不会生效。原因是在同一个类之中,方法互相调用,切面无效 ,而不仅仅是事务。这里事务之所以无效,是因为spring的事务是通过aop实现的。


参考:https://blog.csdn.net/dhklsl/article/details/88354216

相关文章
|
1天前
|
Dubbo Java 应用服务中间件
微服务学习 | Springboot整合Dubbo+Nacos实现RPC调用
微服务学习 | Springboot整合Dubbo+Nacos实现RPC调用
|
1天前
|
Java 开发工具 Maven
根据SpringBoot Guides完成进行示例学习(详细步骤)
根据SpringBoot Guides完成进行示例学习(详细步骤)
6 1
|
1天前
|
监控 安全 Java
Spring cloud原理详解
Spring cloud原理详解
17 0
|
1天前
|
SQL Java 关系型数据库
Spring 事务
Spring 事务
12 1
|
1天前
|
XML 监控 Java
Spring基础 SpringAOP
Spring基础 SpringAOP
12 0
|
1天前
|
存储 前端开发 Java
Spring Boot自动装配的源码学习
【4月更文挑战第8天】Spring Boot自动装配是其核心机制之一,其设计目标是在应用程序启动时,自动配置所需的各种组件,使得应用程序的开发和部署变得更加简单和高效。下面是关于Spring Boot自动装配的源码学习知识点及实战。
15 1
|
1天前
|
Java 数据库连接 数据库
Spring事务简介,事务角色,事务属性
Spring事务简介,事务角色,事务属性
18 2
|
1天前
|
Java 开发者 微服务
Spring Cloud原理详解
【5月更文挑战第4天】Spring Cloud是Spring生态系统中的微服务框架,包含配置管理、服务发现、断路器、API网关等工具,简化分布式系统开发。核心组件如Eureka(服务发现)、Config Server(配置中心)、Ribbon(负载均衡)、Hystrix(断路器)、Zuul(API网关)等。本文讨论了Spring Cloud的基本概念、核心组件、常见问题及解决策略,并提供代码示例,帮助开发者更好地理解和实践微服务架构。此外,还涵盖了服务通信方式、安全性、性能优化、自动化部署、服务网格和无服务器架构的融合等话题,揭示了微服务架构的未来趋势。
35 6
|
1天前
|
Java 数据库连接 数据库
16:事务-Java Spring
16:事务-Java Spring
29 5
|
1天前
|
消息中间件 Java 关系型数据库
Spring事务与分布式事务
这篇文档介绍了事务的概念和数据库事务的ACID特性:原子性、一致性、隔离性和持久性。在并发环境下,事务可能出现更新丢失、脏读和不可重复读等问题,这些问题通过设置事务隔离级别(如读未提交、读已提交、可重复读和序列化)来解决。Spring事务传播行为有七种模式,影响嵌套事务的执行方式。`@Transactional`注解用于管理事务,其属性包括传播行为、隔离级别、超时和只读等。最后提到了分布式事务,分为跨库和跨服务两种情况,跨服务的分布式事务通常通过最终一致性策略,如消息队列实现。