声明式事务原理,传播机制,事务失效情况
声明式事务是一种通过配置而不是编程方式来管理事务的方法,它的原理是基于代理机制和AOP(面向切面编程)实现的。Spring 提供了声明式事务管理的支持,使得开发者可以通过配置文件或注解来定义事务规则,而不必在业务逻辑代码中编写事务管理代码。以下是声明式事务的原理、传播机制和一些事务失效情况的概述:
声明式事务的原理:
Spring 使用代理模式创建了一个代理对象,该代理对象包装了被事务管理的目标对象。
在方法调用前后,Spring 切入了事务管理相关的逻辑,使得事务管理逻辑与业务逻辑相分离。
事务管理逻辑包括开启事务、提交事务、回滚事务等操作,这些操作可以通过配置文件或注解来定义。
如果方法成功执行,事务会被提交;如果方法出现异常,事务会被回滚。
事务传播机制:
Spring 定义了多种事务传播行为,用于控制事务在不同方法之间的传播和交互方式。以下是常见的事务传播行为:
PROPAGATION_REQUIRED:如果当前存在事务,则加入该事务,如果不存在事务,则创建一个新的事务。
PROPAGATION_REQUIRES_NEW:创建一个新的事务,如果当前存在事务,则将其挂起。
PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务,如果不存在事务,则以非事务方式执行。
PROPAGATION_NOT_SUPPORTED:以非事务方式执行,如果当前存在事务,则将其挂起。
PROPAGATION_MANDATORY:要求当前存在事务,如果不存在事务,则抛出异常。
PROPAGATION_NEVER:要求当前不存在事务,如果存在事务,则抛出异常。
PROPAGATION_NESTED:嵌套事务,如果当前存在事务,则在嵌套事务中执行,可以嵌套多层事务。
事务失效情况:
不受检查异常:Spring 默认情况下只对受检查异常(继承自 Exception)进行事务回滚。如果业务代码抛出了不受检查异常(继承自 RuntimeException,如 NullPointerException),事务可能不会回滚,除非显式配置回滚策略。
在同一个类中调用事务方法:在同一个类中的一个事务方法调用另一个事务方法,事务可能不会按预期回滚。这是因为 Spring 使用基于代理的事务管理,事务方法调用必须通过代理对象才能被拦截和管理。
在同一个类中的内部方法不生效:如果在同一个类中的一个非 public 方法(如 private 或 protected)上应用 @Transactional 注解,事务可能不会生效,因为 Spring 使用 AOP 来管理事务,只有外部调用才会被代理拦截。
不同类中的方法互相调用:如果一个被事务管理的方法在不同的类中,并且一个方法调用另一个方法,事务可能不会按预期生效,因为事务是在方法级别而不是对象级别管理的。
RuntimeException 的处理:如果在事务方法中捕获了 RuntimeException 并没有重新抛出,事务可能不会回滚。Spring 只会对未捕获的异常触发回滚。
要避免事务失效情况,应该确保了解声明式事务的工作原理,并按照最佳实践进行配置和使用。此外,要谨慎处理异常,以确保事务按照预期进行回滚。