为了账号安全,请及时绑定邮箱和手机立即绑定

Spring之路(41)–注意规避@Transactional声明式事务失效的情况

标签:
Spring

注意!

Spring中使用@Transactional的声明式事务是足够简单了,对底层逻辑进行了封装,开发人员拿来即用,方便快捷。

但是一定要注意在某些情况下,声明式事务会失效,事务是如此重要,一旦失效可能会带来灾难性后果,所以本篇我们来实验下。

抛出检查型异常时事务失效

首先了解下一场类型:

  • Exception,受检查的异常,在程序中必须使用try…catch进行处理,遇到这种异常必须进行catch或throw,如果不处理,编译器会报错。例如IOException。
  • RuntimeException:非受检查的异常,可以不使用try…catch进行处理。例如常见的NullPointerException。

在我们的观念中,只要有异常,事务就应该回滚,实际上使用@Transactional时,默认只对非受检查异常回滚。例如:

	@Transactional
	public void addTwoBlog() throws Exception{
		BlogDo blog = new BlogDo();
		blog.setContent("测试");
		blogDao.insert(blog);
		blogDao.insert(blog);
		throw new RuntimeException();//非检查异常,回滚
	}
	@Transactional
	public void addTwoBlog() throws Exception{
		BlogDo blog = new BlogDo();
		blog.setContent("测试");
		blogDao.insert(blog);
		blogDao.insert(blog);
		int a=1/0;//非受检查异常,回滚
	}
	@Transactional
	public void addTwoBlog() throws Exception{
		BlogDo blog = new BlogDo();
		blog.setContent("测试");
		blogDao.insert(blog);
		blogDao.insert(blog);
		throw new Exception();//注意!此处为受检查的异常,报错但不会回滚
	}

OK,那么按正常情况下,我们认为一旦有异常,都应该回滚,此时只需要为注解添加rollbackFor=Exception.class属性即可。
例如:

    @Transactional(rollbackFor=Exception.class)//只要抛出异常就会回滚
	public void addTwoBlog() throws Exception{
		BlogDo blog = new BlogDo();
		blog.setContent("测试");
		blogDao.insert(blog);
		blogDao.insert(blog);
		throw new Exception();
	}

一个事务方法调用另一个事务方法时失效

看如下案例:

	@Transactional
	public void startTransaction() throws Exception{
		this.addTwoBlog();
	}
	
	@Transactional(rollbackFor=Exception.class)
	public void addTwoBlog() throws Exception{
		BlogDo blog = new BlogDo();
		blog.setContent("测试");
		blogDao.insert(blog);
		blogDao.insert(blog);
		throw new Exception();
	}

startTransaction和addTwoBlog方法都是事务方法,且这两个方法事务特性不同(一个没有rollbackFor=Exception.class),如果我们调用startTransaction方法,则addTwoBlog中的事务并不会生效。

也就是说,如果在同一个bean中,一个事务方法调用另一个事务方法,可能会导致被调用的事务方法的事务失效!

这是因为Spring的声明式事务使用了代理,具体机制此处不再探讨,但是一定要规避这种事务失效的场景。我们可以通过将对数据库的操作放到一个bean方法里来解决这个问题。

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消