事务
Springboot有3种技术方式来实现让加了@Transactional的方法能使用数据库事务,分别是"动态代理(运行时织入)"、“编译期织入”和“类加载期织入”。这3种技术都是基于AOP(Aspect Oriented Programming,面向切面编程)思想。(在网上看了很多文章,大家伙儿都把AOP称之为一种技术,其实不然,AOP并不特指一种技术,而是一种编程范式,基于AOP编程范式,不同的编程语言都有自己的实现。)
在开发中,尤其要针对多表操作的时候,要注意事物的原子性,举个例子:
在转账模块,A向B转账100元,表设计为A与B为同一个表,但是A转账之后要将A的金额-100,B的金额+100。
这时候如果不考虑原子性的话,那么更新A的数据成功了,而更新B表失败了,就会导致A的钱已经少了,B却没有收到,这是绝对不允许的,因此需要需要保证「转账」的原子性,「要么一起成功,要么一起失败」。使用@Transactional
注解就是专门做这个的。
Transaction 方式(基于动态代理支持)
依赖
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>version</version>
<!-- 5.2.7.RELEASE -->
</dependency>
注解
在启动类上加上@EnableTransactionManagement
注解:
@SpringBootApplication
@EnableTransactionManagement
public class TransactionApplication {
public static void main(String[] args) {
SpringApplication.run(TransactionApplication.class, args);
}
}
在需要使用到事务的地方加上@Transactional
注解:
@Transactional
@GetMapping("insertExceptionT")
public void insert3(){
User u1 = new User();
u1.setUsername("王五");
u1.setPassword("123");
userMapper.insert(u1);
throw new RuntimeException("测试");
}
测试
在测试代码中我们尝试向User表中插入User,分别创建两个接口,一个使用@Transactional
注解,一个不使用,并且在两个接口运行的时候都抛出@RuntimeException
,查看是否会成功。
@GetMapping("insertException")
public void insert2(){
User u1 = new User();
u1.setUsername("李四");
u1.setPassword("123");
userMapper.insert(u1);
throw new RuntimeException("测试");
}
@Transactional
@GetMapping("insertExceptionT")
public void insert3(){
User u1 = new User();
u1.setUsername("王五");
u1.setPassword("123");
userMapper.insert(u1);
throw new RuntimeException("测试");
}
分别访问两个低之后,查看MySQL数据情况:
+----+----------+----------+
| id | username | password |
+----+----------+----------+
| 10 | 李四 | 123 |
+----+----------+----------+
可以发现没有加@Transactional
注解的方法虽然报错了,但是数据依旧插入成功,但是加了注解的方法什么数据都没有插入。
Transactional 失效场景
如果@Transaction
标注的是一个私有方法的话,那么可能会导致IDEA测试的时候事务失效,具体的查看下面这篇文章吧,等有时间的时候再补充一下。
参考文章:「掘金」:SpringBoot事务使用及注意事项