什么是事务?
事务就是为了保证一组数据库操作,要么全部成功,要么全部失败。
SpringBoot中使用事务
spring Boot 使用事务非常简单,首先使用注解 @EnableTransactionManagement 开启事务支持后,然后在访问数据库的Service方法上添加注解 @Transactional 便可。如果在业务类上加@Transactional,则整个类所有方法都支持事务。
在Spring Boot中,当我们使用了spring-boot-starter-jdbc或spring-boot-starter-data-jpa依赖的时候,框架会自动默认分别注入DataSourceTransactionManager或JpaTransactionManager。所以我们不需要任何额外配置就可以用@Transactional注解进行事务的使用。
手动控制事务
当需要在一个方法的内部进行事务控制时,就需要利用spring提供的事务管理器来手动控制。
1、在方法所在的类中注入事务管理器,
2、然后在方法中需要的位置利用事务管理器进行控制:开启事务,提交事务,回滚事务
@Autowired
PlatformTransactionManager platformTransactionManager;
@Autowired
TransactionDefinition transactionDefinition;
public void testTransaction() {
//开启事务
TransactionStatus transactionStatus = platformTransactionManager.getTransaction(transactionDefinition);
try {
CategoryInfo categoryInfo1=new CategoryInfo();
categoryInfo1.setName("分类1");
categoryDataDao.add(categoryInfo1);
//模拟发生异常,事务回滚后应该数据库两条记录都没有
int i=1/0;
CategoryInfo categoryInfo2=new CategoryInfo();
categoryInfo2.setName("分类2");
categoryDataDao.add(categoryInfo2);
//提交事务
platformTransactionManager.commit(transactionStatus);
} catch (Exception e) {
logger.error("发生异常事务回滚");
platformTransactionManager.rollback(transactionStatus);
}
}
事务失效的情况
1、首先我们得清楚事务的实现原理,就是AOP,也就是动态代理。
- 动态代理要求被代理方法是publish,其他修饰不能被代理;
- static修饰的方法属于类,不属于任何对象,也不能被代理;
- final修饰的方法无法重写,也就不能被代理,事务也不会生效。
- 在代理模式下,只拦截通过代理进入的调用方法。直接调用本类方法或使用this调用本类方法,均不是Spring的代理对象,无法实现动态代理,事务也就不会生效。这意味着实际上,自调用目标对象中的一个方法调用目标对象的另一个方法,即使被调用的方法用@Transactional标记,也不会在运行时导致实际的事务。@Transactional在同一个类中调用方法时不起作用。
2、使用try/catch时,只进行捕获,没有抛出异常。
3、Spring事务默认支持RuntimeException运行时异常,抛出的异常为RuntimeException异常及其子类异常事务均可生效;而我们日常常见的异常基本都继承自RuntimeException,所以无需指定异常类型事务也能生效。但若手动抛出Exception异常,或者抛出的自定义异常继承于Exception异常,而Exception是RuntimeException的父类,会导致事务不生效。
解决方式:
(1.指定Spring事务异常捕获类型@Transactional (rollbackFor = Exception.class)。让spring事务 支持Exception异常。
(2.抛出Spring事务支持的异常类型 - RuntimeException或继承于RuntimeException的异常
4、事务有没有开启、传播级别没有设置好