前几天我发了这篇文章《我来出个题:这个事务会不会回滚?》获得了不少不错的反馈,也有很多读者经过微信、群或者邮件的方式,给了我一些关于test4的回复。其中还有直接发给我测试案例,来证实个人答案是错的。今天,咱们就来一块儿看看test4这个争议很大的问题。若是您是刚打开这篇文章,不了解咱们在讨论啥,那能够先点击查看以前的这篇《我来出个题:这个事务会不会回滚?》。经过这两篇文章的解析,相信你会对Spring Data JPA下的事务执行机制有质的飞跃。html
先来讲说,那些写了代码验证"不会回滚"的状况,把这些错误答案的缘由先说清楚,而后再细说test4会回滚的状况。java
根据这两天读者给个人案例或者描述清楚的一些状况,归结了一下,你们写的验证代码之因此不会回滚,主要有如下三个缘由:spring
归家一下出现这些疑问的缘由:没审题和事务基础掌握不牢致使。关于事务基础使用的一些常见注意点,以前写过一篇文章,若是以为这方面知识还不扎实的,建议读一读:《为何加了@Transactional注解,事务没有回滚?》微信
先来看看执行时候报的异常:函数
javax.validation.ConstraintViolationException: Validation failed for classes [com.didispace.chapter310.User] during persist time for groups [javax.validation.groups.Default, ] List of constraint violations:[ ConstraintViolationImpl{interpolatedMessage='个数必须在0和5之间', propertyPath=name, rootBeanClass=class com.didispace.chapter310.User, messageTemplate='{javax.validation.constraints.Size.message}'} ] at org.hibernate.cfg.beanvalidation.BeanValidationEventListener.validate(BeanValidationEventListener.java:140) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.cfg.beanvalidation.BeanValidationEventListener.onPreInsert(BeanValidationEventListener.java:80) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.action.internal.EntityInsertAction.preInsert(EntityInsertAction.java:209) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.action.internal.EntityInsertAction.execute(EntityInsertAction.java:83) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:604) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:478) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.event.internal.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:356) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.event.internal.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:39) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.internal.SessionImpl.doFlush(SessionImpl.java:1454) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.internal.SessionImpl.managedFlush(SessionImpl.java:511) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.internal.SessionImpl.flushBeforeTransactionCompletion(SessionImpl.java:3283) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.internal.SessionImpl.beforeTransactionCompletion(SessionImpl.java:2479) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.engine.jdbc.internal.JdbcCoordinatorImpl.beforeTransactionCompletion(JdbcCoordinatorImpl.java:473) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl.beforeCompletionCallback(JdbcResourceLocalTransactionCoordinatorImpl.java:178) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl.access$300(JdbcResourceLocalTransactionCoordinatorImpl.java:39) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl$TransactionDriverControlImpl.commit(JdbcResourceLocalTransactionCoordinatorImpl.java:271) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final] at org.hibernate.engine.transaction.internal.TransactionImpl.commit(TransactionImpl.java:98) ~[hibernate-core-5.3.7.Final.jar:5.3.7.Final]
这个异常是这个回滚的关键。这个异常javax.validation.ConstraintViolationException
是哪里的呢?还记得之前说的JSR 303不?对的,是Bean Validation中的异常。学习
有的读者说这个不是RuntimeException
,因此不会回滚。很显然,这类判断的都没有实际尝试一下,只要点开源码能够立刻发现,这个异常就是属于RunTimeException
的。测试
实际上,之因此会回滚,与这里使用Spring Data JPA以及Hibernate Validator有直接关系。从JPA 2.0开始,就默认支持了这些Bean Validation的实现,它提供了实体生命周期中pre-persist
, pre-update
,pre-remove
三个事件发生时来执行校验的功能。而在校验的时候,当校验失败,抛出javax.validation.ConstraintViolationException
时,当前事务就会被标记为rollback
。this
要想了解,这其中到底发生了什么,跟踪源码是最好的方式。那么源码从哪里开始看呢?从异常日志中找线索吧。spa
从异常栈中找到最近的一个错误,点开看看。hibernate
错误行数在532行tx.commit()
,习惯性的加上断点,这样下一次进来的时候能够看看当前状况下的各类参数状况。
同时看到下面还有个catch,既然532行出错了,那这里确定会进,因此也加个端点,到时候能够进去看看。
执行程序,调用一下test4,执行到532行,而后进入下一步,看看会到哪里?
这个时候,会进入到org.hibernate.engine.transaction.internal.TransactionImpl
,具体位置以下:
仍是习惯性的,在下面两行重要位置加上断点,以便下次能够快速到这里。
继续按上看的步骤尝试下去,能够来到下图的位置:
能够看到校验异常是从271行出来的,结合278行和280行,是否是清楚这里回滚的缘由了呢?
实践出真知,当你以为困惑的时候,不如动手写一写,调一调,不少答案就能天然浮现!
若是对于test4会回滚还不够理解,或者你还有其余事务执行不如预期的读者,那就跟着个人思路,一步步尝试一下,能够观察的更深刻一些,你对这部分逻辑的理解就更全面了。咱们正在组建高质量的Spring技术交流群,欢迎各类热爱技术的开发者加入参与讨论。这里的每一个人都有本身的闪光点,互相学习,取长补短,长期坚持,愿你们都会成为本身领域里的佼佼者!
欢迎关注个人公众号:程序猿DD,分享其余地方看不到的知识与思考