Spring中的事务实现、失效场景即AOP的相关概念理解

本文介绍了Spring中声明式事务的实现原理,通过AOP完成方法的拦截。讨论了导致事务失效的几种情况,如异常处理、检查异常和方法访问权限,并展示了如何使用AOP进行系统操作日志记录。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

spring实现事务(声明式事务)的本质就是aop完成的,它会对方法的前后进行拦截,在执行方法之前开启事务,在执行完目标方法之后根据执行情况提交或回滚事务。aop就是面向切面编程,在spring中将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取公共模块复用,降低耦合度,事务处理就是其中一项重要应用。事项spring中的事务有两种方式编程式事务和声明式事务,其中编程式事务对业务代码有侵入性,项目中很少使用,应用更为广泛的是声明式事务。

spring中的事务在下面一些情况下会失效:

1.如果方法上自己处理捕获了目标的异常,没有抛出,就会导致事务失效。(原因:事务通知只有捕捉到了目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理掉异常,事务通知无法知悉)如:@Transactional注解标注的方法中try{有异常的代码}catch(Exception e){e.printStackTrace();}。解决方法:在catch块添加throw new RuntimeException(e)抛出。

2.方法抛出检查异常,报错也会导致事务失效。如:@Transactional注解标注的方法throws IOException(或其实现类如FileNotFoundException)(原因:Spring默认只会回滚非检查异常).解决方法:配置rollbackFor属性,如注解写为@Transactional(rollbackFor=Exception.class)

3.方法上不是public修饰的,也会导致事务失效(原因:spring为方法创建代理、添加事务通知前提条件都是方法是public的),解决方法:方法修饰词改成public。

AOP除了事务处理外还可以用来记录系统的操作日志。主要思路:使用aop中的环绕通知,用切点表达式找到记录日志的方法,也可以自定义注解(@annotation:指定一个注解,凡是标有此注解的方法都是切点,更推荐),然后通过环绕通知的参数获取请求方法的参数,比如类名、方法名、参数、返回值、异常信息、日志结果等,获取到这些参数后,保存到数据库

@Aspect
@Component
public class LoggingAspect {
 
    private static final Logger logger = LoggerFactory.getLogger(LoggingAspect.class);

    @Pointcut("execution(* com.example.account.*.*(..))")
    public void accountOperation() {}

    @Before("accountOperation()")
    public void logBefore(JoinPoint joinPoint) {
        logger.info("Starting {} at {}", joinPoint.getSignature().getName(), LocalDateTime.now());
    }

    @AfterReturning(pointcut = "accountOperation()", returning = "result")
    public void logAfter(JoinPoint joinPoint, Object result) {
        logger.info("Finished {} at {} with result {}", joinPoint.getSignature().getName(), LocalDateTime.now(), result);
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值