赞
踩
某一个事务的回滚, 导致另外一个事务已更新的数据丢失了。【$T1 到 T6 $ 可能只有1ms 甚至更短】
某一个事务的提交, 导致另外一个事务已更新的数据丢失了。
某一个事务, 读取了另外一个事务未提交的数据。
某一个事务, 对同一个数据前后读取的结果不一致。
某一个事务, 对同一个表前后查询到的行数不一致。
从上到下,隔离级别由低到高。Y 表示会出现该类问题。N 表示不会【当然这也对应着性能的下降】
通常中间两个是在应用中使用较多的
悲观锁(数据库)
共享锁(S锁)
事务A对某数据加了共享锁后,其他事务只能对该数据加共享锁,但不能加排他锁。
排他锁(X锁)
事务A对某数据加了排他锁后,其他事务对该数据既不能加共享锁,也不能加排他锁。
乐观锁(自定义)
版本号、时间戳等
在更新数据前,检查版本号是否发生变化。若变化则取消本次更新,否则就更新数据(版本号+1)。
Spring 使用一套API 即可面向所有种类的数据库
代码演示:
【需求,在一个用户完成注册时,自动发布一条“新人报道” 的帖子】
@Autowired
private UserMapper userMapper;
@Autowired
private DiscussPostMapper discussPostMapper;
// REQUIRED: 支持当前事务(外部事务),如果不存在则创建新事务.【一定需要事务】
// REQUIRES_NEW: 创建一个新事务,并且暂停当前事务(外部事务).
// NESTED: 如果当前存在事务(外部事务),则嵌套在该事务中执行(独立的提交和回滚),否则就会REQUIRED一样.
@Transactional(isolation = Isolation.READ_COMMITTED, propagation = Propagation.REQUIRED)
public Object save1() {
// 新增用户
User user = new User();
user.setUsername("alpha");
user.setSalt(CommunityUtil.generateUUID().substring(0, 5));
user.setPassword(CommunityUtil.md5("123" + user.getSalt()));
user.setEmail("alpha@qq.com");
user.setHeaderUrl("http://images.nowcoder.com/head/972t.png");
user.setCreateTime(new Date());
userMapper.insertUser(user);
// 新增帖子
DiscussPost post = new DiscussPost();
post.setUserId(user.getId());
post.setTitle("Hello");
post.setContent("新人报道!");
post.setCreateTime(new Date());
discussPostMapper.insertDiscussPost(post);
Integer.valueOf("abc"); // 故意报个错误出来
return "ok";
}
测试一下
package com.nowcoder.community;
import com.nowcoder.community.service.AlphaService;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringRunner;
/**
* @Projectname: community
* @Classname: TransactionTests
* @Author: Ding Jiaxiong
* @Date:2023/4/19 10:53
*/
@RunWith(SpringRunner.class)
@SpringBootTest
@ContextConfiguration(classes = CommunityApplication.class)
public class TransactionTests {
@Autowired
private AlphaService alphaService;
@Test
public void testSave1() {
Object obj = alphaService.save1();
System.out.println(obj);
}
}
现在的save1方法中是有一个错误的,理想运行结果是 事务回滚,不执行插入操作,
运行结果
OK,报错了,但是我更关心的是 数据有没有插入成功
验证了,没有,事务回滚了
这是注解式,接下来演示编程式事务:
@Autowired
private TransactionTemplate transactionTemplate;
public Object save2() {
transactionTemplate.setIsolationLevel(TransactionDefinition.ISOLATION_READ_COMMITTED);
transactionTemplate.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
return transactionTemplate.execute(new TransactionCallback<Object>() {
@Override
public Object doInTransaction(TransactionStatus status) {
// 新增用户
User user = new User();
user.setUsername("beta");
user.setSalt(CommunityUtil.generateUUID().substring(0, 5));
user.setPassword(CommunityUtil.md5("123" + user.getSalt()));
user.setEmail("beta@qq.com");
user.setHeaderUrl("http://image.nowcoder.com/head/999t.png");
user.setCreateTime(new Date());
userMapper.insertUser(user);
// 新增帖子
DiscussPost post = new DiscussPost();
post.setUserId(user.getId());
post.setTitle("你好");
post.setContent("我是新人!");
post.setCreateTime(new Date());
discussPostMapper.insertDiscussPost(post);
Integer.valueOf("abc");
return "ok";
}
});
}
同样测试一下
@Test
public void testSave2() {
Object obj = alphaService.save2();
System.out.println(obj);
}
运行结果
OK,和注解式事务运行效果一样,看看数据库
肯定都是没有插入成功的,出现错误,回滚了
后面就是真正的运用到我们的项目了,注意优先使用注解式事务,如果运行过程中需要对一些东西进行控制,再使用编程式事务。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。