try catch 与事务和程序执行逻辑

1.try catch 与事务的关系

事务在开发中是一个非常重要的环节,try catch 在一定条件下会让事务失效,回滚失败。下面就用代码来测试一下try catch 与事务的关系。

首先准备一张数据库表,原始数据为空,如下:
在这里插入图片描述
然后我们下面的代码都只需要调用一下 learnTryCatch 方法。

1.1 不加try catch正常回滚

看service层代码:

@Service
public class TestMyServiceImpl implements TestMyService {

    @Autowired
    private TestMyMapper testMyMapper;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        TestMy data = getData();
        testMyMapper.insert(data);
        throw new RuntimeException();
    }

    private TestMy getData(){
        TestMy testMy = new TestMy();
        testMy.setName("陆雪琪");
        testMy.setNumber(18);
        testMy.setTime(LocalDateTime.now());
        return testMy;
    }

}

执行之后的结果如下图,可见事务正常回滚。
在这里插入图片描述

1.2 加try catch,事务不能正常回滚

再来看以下代码:

@Service
public class TestMyServiceImpl implements TestMyService {

    @Autowired
    private TestMyMapper testMyMapper;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        TestMy data = getData();
        testMyMapper.insert(data);
        try {
        	data.setId(null);
            data.setName("陆雪琪");
            testMyMapper.insert(data);
            throw new IndexOutOfBoundsException();
        }catch (Exception e){
            e.printStackTrace();
        }  
    }

    private TestMy getData(){
        TestMy testMy = new TestMy();
        testMy.setName("碧瑶");
        testMy.setNumber(18);
        testMy.setTime(LocalDateTime.now());
        return testMy;
    }

}

执行后结果如下图:
在这里插入图片描述
可见,在try中抛出了异常,但是整个方法的事务并没有回滚。

1.3 加try catch,发生异常可以回滚

来看下面的代码:

@Service
public class TestMyServiceImpl implements TestMyService {

    @Autowired
    private TestMyMapper testMyMapper;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        TestMy data = getData();
        testMyMapper.insert(data);
        try {
            data.setId(null);
            data.setName("崔莺莺");
            testMyMapper.insert(data);
            throw new IndexOutOfBoundsException();
        }catch (NullPointerException e){
            e.printStackTrace();
        }
    }

    private TestMy getData(){
        TestMy testMy = new TestMy();
        testMy.setName("冯小青");
        testMy.setNumber(18);
        testMy.setTime(LocalDateTime.now());
        return testMy;
    }

}

这里抛出IndexOutOfBoundsException,但是只捕捉NullPointerException。来看执行结果:
在这里插入图片描述
可见整个方法的事务正常回滚了。

1.4 在try中调用加事务的方法

我们先来做一个测试,不加try catch 调用别的插入方法,看代码:

@Service
public class TestMyServiceImpl implements TestMyService {

    @Autowired
    private TestMyMapper testMyMapper;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        TestMy data = getData();
        testMyMapper.insert(data);
        //调用别的插入数据的方法
        this.testTran();
        throw new IndexOutOfBoundsException();
    }

    @Override
    public void testTran() {
        TestMy data = getData();
        data.setName("苏小小");
        testMyMapper.insert(data);
    }

    private TestMy getData() {
        TestMy testMy = new TestMy();
        testMy.setName("冯小青");
        testMy.setNumber(18);
        testMy.setTime(LocalDateTime.now());
        return testMy;
    }

}

结果如下图,没有新插入数据,可见只要是本方法(加了事务注解)调用的方法,事务都会生效。
在这里插入图片描述
然后我们试一下在try中调用加事务注解的方法,话不多说,看代码:

@Service
public class TestMyServiceImpl implements TestMyService {

    @Autowired
    private TestMyMapper testMyMapper;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        TestMy data = getData();
        testMyMapper.insert(data);
        try {
            data.setId(null);
            data.setName("崔莺莺");
            testMyMapper.insert(data);
            //调用加事务的方法
            this.testTran();
            throw new IndexOutOfBoundsException();
        }catch (Exception e){
            e.printStackTrace();
        }
    }

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void testTran() {
        TestMy data = getData();
        data.setName("苏小小");
        testMyMapper.insert(data);
    }

    private TestMy getData(){
        TestMy testMy = new TestMy();
        testMy.setName("冯小青");
        testMy.setNumber(18);
        testMy.setTime(LocalDateTime.now());
        return testMy;
    }

}

我们知道,learnTryCatch 这个方法的事务是回滚不了的。因为try中的异常被catch捕获了。那么testTran方法被try中的代码调用,事务能回滚吗?我们执行一下看结果:
在这里插入图片描述
从结果可知,冯小青、崔莺莺、苏小小都加上了,说明任何事物都没有生效。

1.5 使用 throws Exception

看代码:

@Service
public class TestMyServiceImpl implements TestMyService {

    @Autowired
    private TestMyMapper testMyMapper;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() throws Exception{
        TestMy data = getData();
        testMyMapper.insert(data);
        //调用插入方法
        this.testTran();
        throw new IndexOutOfBoundsException();
    }

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void testTran() {
        TestMy data = getData();
        data.setName("萧炎");
        testMyMapper.insert(data);
    }

    private TestMy getData() {
        TestMy testMy = new TestMy();
        testMy.setName("柳七月");
        testMy.setNumber(18);
        testMy.setTime(LocalDateTime.now());
        return testMy;
    }

}

我们改用throws 异常来处理异常,事务会回滚吗?话不多说看执行结果:
在这里插入图片描述
可见事务正常回滚。

1.6 结论

从以上案例可知,当try中的异常被catch捕捉时,所有事务无法回滚,未被catch捕获时事务可以正常回滚。
所以这里告诉我们:

  • 1.不要在try中乱抛异常!!!!
  • 2.使用事务就throws Exception ,不要 try catch 。

2 try catch 程序执行逻辑

2.1 程序抛出异常继续执行

来看下面的代码:

@Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        try {
            throw new NullPointerException();
        }catch (Exception e){
            e.printStackTrace();
        }
        System.out.println("抛出了异常!!");
    }

执行结果:
在这里插入图片描述
可见,抛出了异常,还在继续执行代码。

2.2 程序抛出异常终止执行

看代码:

 @Override
    @Transactional(rollbackFor = Exception.class)
    public void learnTryCatch() {
        try {
            throw new IndexOutOfBoundsException();
        }catch (NullPointerException e){
            e.printStackTrace();
        }
        System.out.println("抛出了异常!!");
    }

这里catch没有捕捉try中抛出的异常,程序直接终止:
在这里插入图片描述

2.3 结论

可知,当try中的异常被catch捕获,则程序会继续执行。若try中的异常未被catch捕获,则程序终止执行。
所以别在try catch 中乱抛异常,不然会很奇怪抛出了异常怎么程序还是执行了。

  • 26
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Spring Boot中,你可以使用try-catch语句来处理异常并实现事务的回滚。通常情况下,如果在事务中抛出异常,Spring会自动回滚该事务。但是,如果你想在try-catch块中捕获异常并手动控制事务的回滚,你可以按照以下步骤操作: 1. 确保你的应用程序的事务管理器已经配置好。你可以通过在Spring Boot的配置类上添加`@EnableTransactionManagement`注解来启用事务管理。 2. 在需要进行事务管理的方法上添加`@Transactional`注解。这将告诉Spring该方法需要在事务内执行。 3. 在try-catch块中捕获可能抛出的异常。你可以在catch块中编写逻辑来处理异常,例如记录日志或发送警报。 4. 如果你想手动回滚事务,可以使用`TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()`方法来标记当前事务为回滚状态。 下面是一个示例代码: ```java @Service public class UserService { @Autowired private UserRepository userRepository; @Transactional public void updateUser(User user) { try { // 业务逻辑 userRepository.save(user); } catch (Exception e) { // 异常处理逻辑 // 记录日志或发送警报 // 手动回滚事务 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); } } } ``` 在上面的例子中,`updateUser`方法使用了`@Transactional`注解来启用事务管理。如果在`userRepository.save(user)`方法执行过程中抛出异常,catch块中的逻辑将会被执行,并且事务会被手动回滚。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值