事务嵌套,不回滚的问题描述:
要想事务起作用,必须是主方法名上有@Transactional注解,方法体内不能用try catch;如果用try catch,则catch中必须用throw new RuntimeException();
@Transactional注解应该只被应用到public方法上,不要用在protected、private等方法上,即使用了也将被忽略,不起作用。这是由Spring AOP决定的。
只有来自外部的方法调用才会呗AOP代理捕捉,类内部方法调用类内部的其他方法,子方法并会不引起事务行为,即使被调用的方法上使用有@Transactional注解。
类内部方法调用内部的其他方法,被调用的方法体中如果有try catch,则catch中必须用throw new RuntimeException(),否则即使主方法上加上@Transactional注解,如果被调用的子方法出错也不会抛出异常,不会引起事务起作用。
一、不同类中的方法的事务嵌套问题:
不同类中有@Transactional 的方法调用有@Transactional 的方法
二、同一类中的方法的事务嵌套
同一类中的其他没有@Transactional 注解的方法内部调用有@Transactional 注解的方法,有@Transactional 注解的方法的事务被忽略,不会发生回滚。
原因:spring的事务处理底层是通过CGlib动态代理来实现的。目标方法由外部调用才由 Spring 生成的代理对象来管理,这会造成自调用问题。
为了解决事务嵌套这个问题,我们用到了@AspectJ 面向切面来实现,不能使用spring aop动态代理
AOP的应用场景:
权限控制、缓存控制、事务控制、审计日志、性能监控、分布式追踪、异常处理
Spring AOP的使用方式包含XML配置(不推荐)和注解方式,下面看一下基于注解方式的AOP。
AOP的注解主要包括@Aspect、@Pointcut、Advice三种。在详细记录前先给出一个AOP的简单使用代码(流程:引入依赖–>定义切面类–>在切面类中定义Advice以及前后逻辑),如果要获取方法的一些属性(比如方法名,返回值、参数等等),除了环绕通知需要传入ProceedingJoinPoint对象,其他的Advice则是需要传入JoinPoint对象,两类不同!
1.@Aspect
主要用来标注Java类,表明它是一个切面配置的类,通常下面也会加上@Component注解来表明它由Spring管理
2.@Pointcut
主要有pointCutExpression(切面表达式)来表达,用来描述你要在哪些类的哪些方法上注入代码。其中切面表达式包含了designators(指示器,主要描述通过哪些方式去匹配Java类的哪些方法:如execution()等)和wildcards(通配符:如*)以及operators(运算符:如&&、||、!),具体