欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解MyBatis的SqlSession獲取流程

 更新時間:2023年07月10日 10:15:58   作者:半夏之沫  
SqlSession的獲取是通過SqlSessionFactory的openSession() 方法,那么具體的獲取流程是什么,所以本文就給大家詳細講解一下MyBatis的SqlSession獲取流程,需要的朋友可以參考下

前言

已知在MyBatis的使用中,使用MyBatis時會先讀取配置文件mybatis-config.xml為字符流或者字節(jié)流,然后通過SqlSessionFactoryBuilder基于配置文件的字符流或字節(jié)流來構建SqlSessionFactory,然后再通過SqlSessionFactoryopenSession() 方法獲取SqlSession,示例代碼如下所示。

public static void main(String[] args) throws Exception {
    String resource = "mybatis-config.xml";
    SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder()
            .build(Resources.getResourceAsStream(resource));
    SqlSession sqlSession = sqlSessionFactory.openSession();
}

上述示例代碼中的SqlSessionFactory實際為DefaultSqlSessionFactory,本篇文章將對DefaultSqlSessionFactory獲取SqlSession的流程進行學習。

正文

DefaultSqlSessionFactoryopenSession() 方法如下所示。

public SqlSession openSession() {
    return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false);
}

繼續(xù)看openSessionFromDataSource() 方法的實現(xiàn),如下所示。

private SqlSession openSessionFromDataSource(ExecutorType execType, 
                                             TransactionIsolationLevel level, 
                                             boolean autoCommit) {
    Transaction tx = null;
    try {
        // Environment中包含有事務工廠和數(shù)據(jù)源
        final Environment environment = configuration.getEnvironment();
        // 獲取事務工廠
        // 如果配置了JDBC事務,則獲取JDBC事務工廠,否則獲取MANAGED事務工廠
        final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
        // 創(chuàng)建事務
        tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
        // 創(chuàng)建執(zhí)行器
        final Executor executor = configuration.newExecutor(tx, execType);
        // 構建DefaultSqlSession并返回
        return new DefaultSqlSession(configuration, executor, autoCommit);
    } catch (Exception e) {
        closeTransaction(tx);
        throw ExceptionFactory.wrapException("Error opening session.  Cause: " + e, e);
    } finally {
        ErrorContext.instance().reset();
    }
}

openSessionFromDataSource() 方法會先從Configuration中將Environment獲取出來,Environment中包含有事務工廠和數(shù)據(jù)源,Environment是在MyBatis配置文件中進行配置并會在加載MyBatis配置文件時被添加到Configuration中。

MyBatis的配置文件中可以配置兩種事務工廠,為JDBC事務工廠和MANAGED事務工廠,它們分別可以創(chuàng)建JDBC事務和MANAGED事務,類圖如下所示。

獲取事務工廠的getTransactionFactoryFromEnvironment() 方法如下所示。

private TransactionFactory getTransactionFactoryFromEnvironment(Environment environment) {
    if (environment == null || environment.getTransactionFactory() == null) {
        // 配置了事務工廠則使用配置的事務工廠
        // MyBatis可以配置JDBC和MANAGED這兩種事務工廠
        return new ManagedTransactionFactory();
    }
    // 沒做配置的情況下使用MANAGED事務工廠
    return environment.getTransactionFactory();
}

MyBatis中如果使用JDBC事務,則事務的管理由java.sql.Connection完成,如果使用MANAGED事務,則MyBatis會將事務的管理交由WEB容器(Tomcat,JBoss等)來完成。以commit()rollback() 方法為例,看下JdbcTransactionManagedTransaction對這兩個方法的實現(xiàn),如下所示。

public class JdbcTransaction implements Transaction {
    // ......
    @Override
    public void commit() throws SQLException {
        if (connection != null && !connection.getAutoCommit()) {
            if (log.isDebugEnabled()) {
                log.debug("Committing JDBC Connection [" + connection + "]");
            }
            connection.commit();
        }
    }
    @Override
    public void rollback() throws SQLException {
        if (connection != null && !connection.getAutoCommit()) {
            if (log.isDebugEnabled()) {
                log.debug("Rolling back JDBC Connection [" + connection + "]");
            }
            connection.rollback();
        }
    }
    // ......
}
public class ManagedTransaction implements Transaction {
    // ......
    @Override
    public void commit() throws SQLException {
        // 不做任何事情
    }
    @Override
    public void rollback() throws SQLException {
        // 不做任何事情
    }
    // ......
}

回到openSessionFromDataSource() 方法,創(chuàng)建好事務后,會構建執(zhí)行器Executor,看一下newExecutor(Transaction transaction, ExecutorType executorType) 方法的實現(xiàn),如下所示。

public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
    executorType = executorType == null ? defaultExecutorType : executorType;
    executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
    Executor executor;
    // 根據(jù)ExecutorType的枚舉值創(chuàng)建對應類型的Executor
    if (ExecutorType.BATCH == executorType) {
        executor = new BatchExecutor(this, transaction);
    } else if (ExecutorType.REUSE == executorType) {
        executor = new ReuseExecutor(this, transaction);
    } else {
        executor = new SimpleExecutor(this, transaction);
    }
    // 如果MyBatis配置文件中開啟了二級緩存
    if (cacheEnabled) {
        // 創(chuàng)建CachingExecutor作為Executor的裝飾器,為Executor增加二級緩存功能
        executor = new CachingExecutor(executor);
    }
    // 將插件邏輯添加到Executor中
    executor = (Executor) interceptorChain.pluginAll(executor);
    return executor;
}

在上面構建執(zhí)行器Executor的方法中,主要是做了三件事情,解釋如下。

第一件事情,根據(jù)ExecutorType的枚舉值創(chuàng)建對應類型的Executor,ExecutorType枚舉值如下所示。

public enum ExecutorType {
    SIMPLE, REUSE, BATCH
}

MyBatis提供了三種類型的Executor,分別為BatchExecutor,ReuseExecutorSimpleExecutor,類圖如下所示。

BaseExecutor中定義了四個抽象方法,如下所示。

protected abstract int doUpdate(MappedStatement ms, Object parameter) throws SQLException;
protected abstract List<BatchResult> doFlushStatements(boolean isRollback) throws SQLException;
protected abstract <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql)
    throws SQLException;
protected abstract <E> Cursor<E> doQueryCursor(MappedStatement ms, Object parameter, RowBounds rowBounds, BoundSql boundSql)
    throws SQLException;

BaseExecutor提供的方法中會調用這四個抽象方法,同時這四個抽象方法需要由BatchExecutor,ReuseExecutorSimpleExecutor根據(jù)各自的功能進行實現(xiàn),所以Executor的設計使用了模板設計模式。

第二件事情,如果在MyBatis配置文件中開啟了二級緩存,則為Executor創(chuàng)建裝飾器CachingExecutor,以增加對二級緩存功能的支持。CachingExecutorExecutor的關系如下所示。

CachingExecutor實現(xiàn)了Executor接口,同時CachingExecutor持有Executor的引用,這是裝飾器模式的應用。關于MyBatis中的緩存,會在后續(xù)的文章中進行介紹。

第三件事情,如果配置了插件,則將插件的邏輯添加到Executor中。關于MyBatis中的插件,會在后續(xù)的文章中進行介紹。

回到openSessionFromDataSource() 方法,創(chuàng)建好Executor后,就會調用DefaultSqlSession的構造方法創(chuàng)建一個DefaultSqlSession并返回,在DefaultSqlSession的構造方法中只是進行了簡單的賦值,如下所示。

public DefaultSqlSession(Configuration configuration, Executor executor, boolean autoCommit) {
    this.configuration = configuration;
    this.executor = executor;
    this.dirty = false;
    this.autoCommit = autoCommit;
}

至此,SqlSession就創(chuàng)建出來了。

總結

SqlSession的獲取是通過SqlSessionFactoryopenSession() 方法,SqlSession的創(chuàng)建過程主要先是創(chuàng)建事務管理器,然后基于事務管理器創(chuàng)建執(zhí)行器Executor,最后基于Executor創(chuàng)建SqlSession。

通常,SqlSessionFactoryDefaultSqlSessionFactory,創(chuàng)建出來的SqlSessionDefaultSqlSession。

整體的一個流程圖如下所示。

以上就是詳解MyBatis的SqlSession獲取流程的詳細內容,更多關于MyBatis SqlSession獲取流程的資料請關注腳本之家其它相關文章!

相關文章

  • Mybatis中#{}和${}傳參的區(qū)別及#和$的區(qū)別小結

    Mybatis中#{}和${}傳參的區(qū)別及#和$的區(qū)別小結

    這篇文章主要介紹了Mybatis中#{}和${}傳參的區(qū)別及#和$的區(qū)別小結 的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-07-07
  • Spring?MVC異步上傳、跨服務器上傳和文件下載功能實現(xiàn)

    Spring?MVC異步上傳、跨服務器上傳和文件下載功能實現(xiàn)

    這篇文章主要介紹了Spring?MVC異步上傳、跨服務器上傳和文件下載功能實現(xiàn),本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-07-07
  • java把字符串寫入文件里的簡單方法分享

    java把字符串寫入文件里的簡單方法分享

    這篇文章主要介紹了java把字符串寫入到文件里的簡單方法,這是跟一個外國朋友學的代碼,需要的朋友可以參考下
    2014-03-03
  • SpringBoot整合PostgreSQL的示例代碼

    SpringBoot整合PostgreSQL的示例代碼

    本文主要介紹了SpringBoot整合PostgreSQL的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-07-07
  • Java中的密碼加密方式

    Java中的密碼加密方式

    文章介紹了Java中使用MD5算法對密碼進行加密的方法,以及如何通過加鹽和多重加密來提高密碼的安全性,MD5是一種不可逆的哈希算法,適合用于存儲密碼,因為其輸出的摘要長度固定,且不容易發(fā)生碰撞,此外,通過加鹽和多重加密,可以進一步增加密碼的復雜性和安全性
    2025-01-01
  • java.net.MalformedURLException異常的解決方法

    java.net.MalformedURLException異常的解決方法

    下面小編就為大家?guī)硪黄猨ava.net.MalformedURLException異常的解決方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • Java如何使用poi導入導出excel工具類

    Java如何使用poi導入導出excel工具類

    這篇文章主要介紹了Java如何使用poi導入導出excel工具類問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-06-06
  • 本真的REST架構風格理解

    本真的REST架構風格理解

    這篇文章主要為大家介紹了本真的REST架構風格的深入理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-03-03
  • Java基礎之Web服務器與Http詳解

    Java基礎之Web服務器與Http詳解

    無論你是前端開發(fā)者還是后端開發(fā)者,以及測試工程師,這篇文章的知識都是你需要弄懂的。讀完這一篇文章,將全面弄懂 HTTP 協(xié)議、TCP 協(xié)議,面試官再也難不倒你相關知識
    2021-09-09
  • 關于Java中Object類的幾個方法示例

    關于Java中Object類的幾個方法示例

    這篇文章主要給大家介紹了關于Java中Object類的幾個方法的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-05-05

最新評論