1、未被Spring管理的类中的方法
这种情况是指:没有在类上添加@Service、@Repository、@Component等注解
将类交由Spring管理,然后该类中还有加上了@Transactional注解
例如
@Service //如果没有添加@Service这个注解,就是一种失效的情况
public class MyService {
@Transactional
public void myTransactionalMethod() {
// 事务管理的代码
}
}
2、未正确配置事务管理器
在Spring Boot中,通常会自动配置事务管理器。然而,如果你手动配置了数据源或
事务管理器,并且配置不正确,事务可能会失效。确保你的
DataSource和PlatformTransactionManager配置正确
这里是它的一些错误情况:
1、数据源配置错误: 确保数据源的驱动类、连接URL、用户名和密码等配置
是正确的。
2、事务管理器配置错误: 确保事务管理器的配置正确,并且它引用了正确的
实体管理器工厂。
3、实体管理器工厂配置错误: 确保实体管理器工厂的配置正确,包括数据源的
引用、实体类的扫描路径等。
4、未加入适当的数据库依赖: 确保项目的依赖中包含了正确的数据库
驱动依赖。
3、异常被捕获而不重新抛出
Spring事务默认只对未捕获的运行时异常进行回滚。如果在事务内捕获了异常并
没有重新抛出,事务可能不会回滚。
看如下的代码,这个哪里有问题呢?我们该如何来修改它
@Transactional
public void myTransactionalMethod() {
try {
// 可能会抛出异常的代码
// ...
} catch (Exception e) {
// 异常被捕获,但没有重新抛出
// 可能导致事务不回滚
log.error("An error occurred: {}", e.getMessage());
}
}
在上述代码中,如果在try块中的代码抛出了异常,但在catch块中没有重新抛出
该异常,那么事务可能不会回滚。因为Spring默认只回滚未捕获的运行时异常。
为了确保事务能够正常回滚,应该在catch块中将异常重新抛出或采取其他
适当的措施。
下面是它修改之后的代码:
@Transactional
public void myTransactionalMethod() {
try {
// 可能会抛出异常的代码
// ...
} catch (Exception e) {
// 将异常重新抛出
throw e;
}
}
4、自调用问题
这个就是一般最容易出现的问题了(个人感觉)
就是在同一个类中,直接调用了另一个加上了@Transactional注解的方法,
使Spring可能无法截获这个调用(因为调用并没有经过代理对象),
所以事务失效了。
就是下面例子这种情况:
@Service
public class MyService {
@Transactional
public void methodA() {
// 一些数据库操作
}
public void methodB() {
// 一些逻辑操作
methodA(); // 在同一个类中直接调用methodA
}
}
在这个例子中,methodB在同一个类中直接调用了methodA。Spring的AOP代理没有介入,即没有生成代理对象,所以methodA可能就受不到@Transactional注解的影响。
那如何来解决这种情况呢?
解决方法:
1、使用代理对象调用方法A(通过GPT了解的)
2、使用AopContext.currentProxy()获取当前代理对象(黑马视频中学的)
方法一: 使用代理对象调用方法A
@Service
public class MyService {
@Autowired
private MyService self; // 自动注入当前类的实例
@Transactional
public void methodB() {
// 通过代理对象调用methodA
self.methodA();
}
public void methodA() {
// 一些数据库操作
}
}
方法二:使用AopContext.currentProxy()获取当前代理对象
//使用前提:需要在SpringBoot启动类上加上@EnableAspectJAutoProxy(exposeProxy = true)
//该注解目的是:暴露代理对象,如果没有暴露,我们是拿不到代理对象的
@Service
public class MyService {
@Transactional
public void methodB() {
// 获取当前代理对象
MyService proxy = (MyService) AopContext.currentProxy();
// 通过代理对象调用methodA
proxy.methodA();
}
@Transactional
public void methodA() {
// 一些数据库操作
}
}
更多了解:
https://www.cnblogs.com/xiaowangbangzhu/p/17143288.html