面试官:问你一个,Spring事务是如何传播的?( 三 )
else if (definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRED ||definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRES_NEW ||definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {//先挂起SuspendedResourcesHolder suspendedResources = suspend(null);if (debugEnabled) {logger.debug("Creating new transaction with name [" + definition.getName() + "]: " + definition);}try {boolean newSynchronization = (getTransactionSynchronization() != SYNCHRONIZATION_NEVER);//创建事务状态对象 , 其实就是封装了事务对象的一些信息 , 记录事务状态的DefaultTransactionStatus status = newTransactionStatus(definition, transaction, true, newSynchronization, debugEnabled, suspendedResources);//开启事务,重点看看 DataSourceTransactionObjectdoBegin(transaction, definition);//开启事务后 , 改变事务状态prepareSynchronization(status, definition);return status;}catch (RuntimeException | Error ex) {resume(null, suspendedResources);throw ex;}}
第一次进来时 , PROPAGATION_REQUIRED、PROPAGATION_REQUIRES_NEW和PROPAGATION_NESTED都会进入到这里 , 首先会调用suspend挂起当前存在的事务 , 在这里没啥作用 。 接下来通过newTransactionStatus创建了DefaultTransactionStatus对象 , 这个对象主要就是存储当前事务的一些状态信息 , 需要特别注意newTransaction属性设置为了true , 表示是一个新事务 。 状态对象创建好之后就是通过doBegin开启事务 , 这是一个模板方法:
【面试官:问你一个,Spring事务是如何传播的?】 protected void doBegin(Object transaction, TransactionDefinition definition) {DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;Connection con = null;try {//如果没有数据库连接if (!txObject.hasConnectionHolder() ||txObject.getConnectionHolder().isSynchronizedWithTransaction()) {//从连接池里面获取连接Connection newCon = obtainDataSource().getConnection();if (logger.isDebugEnabled()) {logger.debug("Acquired Connection [" + newCon + "] for JDBC transaction");}//把连接包装成ConnectionHolder , 然后设置到事务对象中txObject.setConnectionHolder(new ConnectionHolder(newCon), true);}txObject.getConnectionHolder().setSynchronizedWithTransaction(true);con = txObject.getConnectionHolder().getConnection();//从数据库连接中获取隔离级别Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);txObject.setPreviousIsolationLevel(previousIsolationLevel);// Switch to manual commit if necessary. This is very expensive in some JDBC drivers,// so we don't want to do it unnecessarily (for example if we've explicitly// configured the connection pool to set it already).if (con.getAutoCommit()) {txObject.setMustRestoreAutoCommit(true);if (logger.isDebugEnabled()) {logger.debug("Switching JDBC Connection [" + con + "] to manual commit");}//关闭连接的自动提交 , 其实这步就是开启了事务con.setAutoCommit(false);}//设置只读事务 从这一点设置的时间点开始(时间点a)到这个事务结束的过程中 , 其他事务所提交的数据 , 该事务将看不见!//设置只读事务就是告诉数据库 , 我这个事务内没有新增 , 修改 , 删除操作只有查询操作 , 不需要数据库锁等操作 , 减少数据库压力prepareTransactionalConnection(con, definition);//自动提交关闭了 , 就说明已经开启事务了 , 事务是活动的txObject.getConnectionHolder().setTransactionActive(true);int timeout = determineTimeout(definition);if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {txObject.getConnectionHolder().setTimeoutInSeconds(timeout);}// Bind the connection holder to the thread.if (txObject.isNewConnectionHolder()) {//如果是新创建的事务 , 则建立当前线程和数据库连接的关系TransactionSynchronizationManager.bindResource(obtainDataSource(), txObject.getConnectionHolder());}}catch (Throwable ex) {if (txObject.isNewConnectionHolder()) {DataSourceUtils.releaseConnection(con, obtainDataSource());txObject.setConnectionHolder(null, false);}throw new CannotCreateTransactionException("Could not open JDBC Connection for transaction", ex);} }
这个方法里面主要做了六件事:
- 首先从连接池获取连接并保存到DataSourceTransactionObject对象中 。
- 关闭数据库的自动提交 , 也就是开启事务 。
- 获取数据库的隔离级别 。
- 根据属性设置该事务是否为只读事务 。
- 将该事务标识为活动事务(transactionActive=true) 。
- 将ConnectionHolder对象与当前线程绑定 。
- 蛋壳公寓|官媒发声:绝不能让“割韭菜者”一跑了之!
- 表达|重磅!2021世界安防博览会官方宣贯会正式召开,百余家企业表达参展意愿
- 公司|LVMH首席数字官跳槽至加密数字钱包公司Ledger
- 成为佛山移动服务体验官 表白留言赢取百元话费
- Store|在BlueMail的App Store反垄断案中 法官作出有利于苹果公司的判决
- 正式|首批体验官正式招募!仰望高端定制,OriginOS玩法大搜罗
- 全新|首批支持5款机型,vivo开启OriginOS首批体验官招募
- 全新|OriginOS招募首批体验官,网友:先冲为敬
- 无人|无人维护?官方打脸:Element UI for Vue 3.0 来了!
- 代言人|OPPO官宣“周冬雨”成为新代言人,周冬雨排列难道会被用上?