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

相关文章
|
5月前
|
搜索推荐 JavaScript Java
基于springboot的儿童家长教育能力提升学习系统
本系统聚焦儿童家长教育能力提升,针对家庭教育中理念混乱、时间不足、个性化服务缺失等问题,构建科学、系统、个性化的在线学习平台。融合Spring Boot、Vue等先进技术,整合优质教育资源,提供高效便捷的学习路径,助力家长掌握科学育儿方法,促进儿童全面健康发展,推动家庭和谐与社会进步。
|
6月前
|
缓存 Java 开发者
【Spring】原理:Bean的作用域与生命周期
本文将围绕 Spring Bean 的作用域与生命周期展开深度剖析,系统梳理作用域的类型与应用场景、生命周期的关键阶段与扩展点,并结合实际案例揭示其底层实现原理,为开发者提供从理论到实践的完整指导。
751 22
|
6月前
|
SQL Java 关系型数据库
Spring事务传播机制:7种姿势教你玩转"事务接力赛"
事务传播机制是Spring框架中用于管理事务行为的重要概念,它决定了在方法调用时事务如何传递与执行。通过7种传播行为,开发者可以灵活控制事务边界,适应不同业务场景。例如:REQUIRED默认加入或新建事务,REQUIRES_NEW独立开启新事务,NESTED支持嵌套回滚等。合理使用传播机制不仅能保障数据一致性,还能提升系统性能与健壮性。掌握这“七种人格”,才能在复杂业务中游刃有余。
|
6月前
|
人工智能 Java 开发者
【Spring】原理解析:Spring Boot 自动配置
Spring Boot通过“约定优于配置”的设计理念,自动检测项目依赖并根据这些依赖自动装配相应的Bean,从而解放开发者从繁琐的配置工作中解脱出来,专注于业务逻辑实现。
2198 0
|
5月前
|
XML Java 测试技术
《深入理解Spring》:IoC容器核心原理与实战
Spring IoC通过控制反转与依赖注入实现对象间的解耦,由容器统一管理Bean的生命周期与依赖关系。支持XML、注解和Java配置三种方式,结合作用域、条件化配置与循环依赖处理等机制,提升应用的可维护性与可测试性,是现代Java开发的核心基石。
|
5月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
619 2
|
7月前
|
Java 关系型数据库 数据库
深度剖析【Spring】事务:万字详解,彻底掌握传播机制与事务原理
在Java开发中,Spring框架通过事务管理机制,帮我们轻松实现了这种“承诺”。它不仅封装了底层复杂的事务控制逻辑(比如手动开启、提交、回滚事务),还提供了灵活的配置方式,让开发者能专注于业务逻辑,而不用纠结于事务细节。
1013 1
|
8月前
|
缓存 安全 Java
Spring 框架核心原理与实践解析
本文详解 Spring 框架核心知识,包括 IOC(容器管理对象)与 DI(容器注入依赖),以及通过注解(如 @Service、@Autowired)声明 Bean 和注入依赖的方式。阐述了 Bean 的线程安全(默认单例可能有安全问题,需业务避免共享状态或设为 prototype)、作用域(@Scope 注解,常用 singleton、prototype 等)及完整生命周期(实例化、依赖注入、初始化、销毁等步骤)。 解析了循环依赖的解决机制(三级缓存)、AOP 的概念(公共逻辑抽为切面)、底层动态代理(JDK 与 Cglib 的区别)及项目应用(如日志记录)。介绍了事务的实现(基于 AOP
292 0
|
8月前
|
监控 架构师 NoSQL
spring 状态机 的使用 + 原理 + 源码学习 (图解+秒懂+史上最全)
spring 状态机 的使用 + 原理 + 源码学习 (图解+秒懂+史上最全)
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
458 9

热门文章

最新文章