MyBatis openSession(),close(),和commit() 底層代碼剖析


一:MyBatis工具類 中openSession到底做了什么?

        Mybatis工具類

 1     private static final String RESOURCE = "mybatis-config.xml";
 2     private static SqlSessionFactory sqlSessionFactory = null;
 3     private static ThreadLocal<SqlSession> threadLocal = new ThreadLocal<SqlSession>();
 4 
 5 
 6 
 7 
 8     //關閉sqlsession
 9     public static void closeSession(){
10         SqlSession session = (SqlSession) threadLocal.get(); // 2
11         threadLocal.set(null);
12         if (session !=null){
13             session.close();
14         }
15     }
16 
17     public static SqlSession getSessionTwo() {
18         //讀取配置文件
19         try {
20             InputStream stream = Resources.getResourceAsStream(RESOURCE);
21             sqlSessionFactory = new SqlSessionFactoryBuilder().build(stream);
22 
23             return sqlSessionFactory.openSession();   //返回openSession 24         } catch (IOException e) {
25             e.printStackTrace();
26         }
27 
28         return null;
29     }

 首先點開openSession 發現踏實sqlsessionFactory的一個方法

 1 package org.apache.ibatis.session;
 2 
 3 import java.sql.Connection;
 4 
 5 public interface SqlSessionFactory {
 6     SqlSession openSession();
 7 
 8     SqlSession openSession(boolean var1);
 9 
10     SqlSession openSession(Connection var1);

 

然后再看這 個方法的實現類DefaultSqlSessionFactory

這里主要返回他自己類的一個方法openSessionFroDataSource()  (數據源) 並賦值 autoCommit 為false

然后 進入這個方法  

 private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {
        Transaction tx = null;

        DefaultSqlSession var8;
        try {
            Environment environment = this.configuration.getEnvironment();
            TransactionFactory transactionFactory = this.getTransactionFactoryFromEnvironment(environment);
            tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
            Executor executor = this.configuration.newExecutor(tx, execType);
            var8 = new DefaultSqlSession(this.configuration, executor, autoCommit);
        } catch (Exception var12) {
            this.closeTransaction(tx);
            throw ExceptionFactory.wrapException("Error opening session.  Cause: " + var12, var12);
        } finally {
            ErrorContext.instance().reset();
        }

        return var8;
    }

 

 
        
可以看到他這個方法主要是初始化一些configure.xml的配置信息和DefaultSqlSession

所以openSession的主要就是初始化了
configure.xml的配置信息和DefaultSqlSession

 二:MyBatis工具類 中sqlSession.close()底層為什么回滾事務?

 首先進入 close()找到他的實現類DefaultsqlSession

   DefaultsqlSession里的close方法

 

  

     

session.close(); 底層為什么可以回滾事務?????
DefaultsqlSession里的close方法
 public void close() {
        try {
            this.executor.close(this.isCommitOrRollbackRequired(false));
            this.dirty = false;
        } finally {
            ErrorContext.instance().reset();
        }

    }
    
 進入Executor接口  查看他的實現類BaseExecutor 找到close()方法
public void close(boolean forceRollback) {   //需要傳入一個boolean參數
        try {
            try {
                this.rollback(forceRollback);   //為true則rollback
            } finally {
                if(this.transaction != null) {   
                    this.transaction.close();
                }

            }
        } catch (SQLException var11) {
            log.warn("Unexpected exception on closing transaction.  Cause: " + var11);
        } finally {
            this.transaction = null;
            this.deferredLoads = null;
            this.localCache = null;
            this.localOutputParameterCache = null;
            this.closed = true;
        }

    }
然后再看下executor.close()里的參數
this.executor.close(this.isCommitOrRollbackRequired(false));
    ------>isCommitOrRollbackRequired
    private boolean isCommitOrRollbackRequired(boolean force) {
        return true || false;
    }
//可以看到如果沒有提交之前調用close() ,isComitOrRollbackRequired()返回的是true
然后Executor的實現類BaseExecutor的close()方法為參數
forceRollback為true 然后進入rollback(forceRollBack)參數為true
看下最終的rollback方法  
public void rollback(boolean required) throws SQLException {
if(!this.closed) {
try {
this.clearLocalCache();
this.flushStatements(true);
} finally {
if(required) {
this.transaction.rollback(); //如果傳入的參數為 true 則進行事務 回滾
}

}
}
 
 
        

 

另外至於為什么沒有提交之前close會回滾事務,提交了之后則是關閉事務

主要在於Executor的實現類BaseExecutor的

isCommitOrRollbackRequired()參數改變了。

三:commit()

    session.commit(); 為什么能提交事務,他的實現和close()的實先基本差不多,主要是

isCommitOrRollbackRequired()方法

     參數的真假,真則提交,假便關閉
    session會話




000000000寫的好low啊自嘲




 

 

 

 

 

 

 

 

 

   


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM