自以为是
在我的想象当中,以为只要给这个方法注释成@Transactional, 就会处理事务,然而并非我想象的那样,今天测试了下事务,记录下
测试
1.首先建立一个JUnit 类来进行测试第一种方式
@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration({"classpath*:/applicationContext. " })public class SpringTransTester { @Resource private UserMoneyMapper userMoney; @Transactional public void doInsert() { UserMoney ins = new UserMoney(); ins.setId("2"); ins.setBalance(100.00); userMoney.insert(ins); ins.setId("2"); // ID相同,会抛出异常 ins.setBalance(99.00); userMoney.insert(ins); } @Test public void testTrans() { List<UserMoney> list = userMoney.qryQuick("yangmf"); doInsert(); System.out.println("testTrans"); }}数据库中插入了一条记录,第二条出现了错误,但是没有进行回滚
百度查了下资料:
通一个类里面如果A方法没有事务,调用带事务B方法,这样B方法的事务会被忽略。2. 改造下,把事务的方法单独放到Service中
// 注意该类放到Spring能扫描到的地方@Servicepublic class SpringTransService { @Resource private UserMoneyMapper userMoney; @Transactional public void doInnerInsert() { UserMoney ins = new UserMoney(); ins.setId("2"); ins.setBalance(100.00); userMoney.insert(ins); ins.setId("2"); // ID相同,会抛出异常 ins.setBalance(99.00); userMoney.insert(ins); } public void doInsert() { doInnerInsert(); }}@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration({"classpath*:/applicationContext. " })public class SpringTransTester { @Autowired private SpringTransService service; @Test public void testTrans() { service.doInsert(); System.out.println("testTrans"); }}这样其实和第一种方法一样,由本类的方法来调用事务, 效果和第一个一样, 但是如果在doInsert上面也加上@Transactional,那么事务会起作用(doInnerInsert,的注解可以去掉)
3. 再改造下
// 注意该类放到Spring能扫描到的地方@Servicepublic class SpringTransService { @Resource private UserMoneyMapper userMoney; @Transactional public void doInsert() { UserMoney ins = new UserMoney(); ins.setId("2"); ins.setBalance(100.00); userMoney.insert(ins); ins.setId("2"); // ID相同,会抛出异常 ins.setBalance(99.00); userMoney.insert(ins); }}@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration({"classpath*:/applicationContext. " })public class SpringTransTester { @Autowired private SpringTransService service; @Test public void testTrans() { service.doInsert(); System.out.println("testTrans"); }}这样可以,插入第二条是出现了一次,记录没有插入进去
4. 再测试下抛出Exception异常
@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration({"classpath*:/applicationContext. " })public class SpringTransTester { @Autowired private SpringTransService service; @Test // 加上抛出异常 public void testTrans() throws Exception { service.doInsert(); System.out.println("testTrans"); }}@Servicepublic class SpringTransService { @Resource private UserMoneyMapper userMoney; // 手动抛出异常 @Transactional public void doInsert() throws Exception { UserMoney ins = new UserMoney(); ins.setId("2"); ins.setBalance(100.00); userMoney.insert(ins); ins.setId("3"); // ID不同相同,会抛出异常 ins.setBalance(99.00); userMoney.insert(ins); throw new Exception("测试事务"); }}异常抛出了,事务没有起作用,数据都存进去了会, 如果把Exception改成RuntimeException,事务会进行回滚,看来异常不能简单的写个Exception。可以继承RuntimeException
继续阅读与本文标签相同的文章
-
MySQL 5.6 全局事务 ID(GTID)实现原理(一)
2026-06-02栏目: 教程
-
Disruptor 全解析(6):为什么它这么快 (二) - 神奇的 cacheline 补齐
2026-06-02栏目: 教程
-
Disruptor 全解析(5):为什么它这么快 (一) - Locks Are Bad
2026-06-02栏目: 教程
-
Disruptor 全解析(4):依赖关系组装
2026-06-02栏目: 教程
-
Disruptor 全解析(2):如何从 Ring Buffer 读取?
2026-06-02栏目: 教程
