摘要:簡介上一篇文章源碼解析一初始化和動態代理分析了解析配置文件以及動態代理相關的源碼,這一篇接著上一篇探究的執行流程,另外了解一下中的緩存。總結本文主要分析了的執行流程,結合上一篇文章基本了解了的運行原理。
簡介
上一篇文章(MyBatis 源碼解析(一):初始化和動態代理)分析了 MyBatis 解析配置文件以及 Mapper 動態代理相關的源碼,這一篇接著上一篇探究 SqlSession 的執行流程,另外了解一下 MyBatis 中的緩存。
openSessionMyBatis 在解析完配置文件后生成了一個 DefaultSqlSessionFactory 對象,后續執行 SQL 請求的時候都是調用其 openSession 方法獲得 SqlSessison,相當于一個 SQL 會話。 SqlSession 提供了操作數據庫的一些方法,如 select、update 等。
先看一下 DefaultSqlSessionFactory 的 openSession 的代碼:
public SqlSession openSession() { return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false); } private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) { Transaction tx = null; try { // 從 configuration 取出配置 final Environment environment = configuration.getEnvironment(); final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment); tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit); // 每個 SqlSession 都有一個多帶帶的 Executor 對象 final Executor executor = configuration.newExecutor(tx, execType, autoCommit); // 返回 DefaultSqlSession 對象 return new DefaultSqlSession(configuration, executor); } catch (Exception e) { closeTransaction(tx); // may have fetched a connection so lets call close() throw ExceptionFactory.wrapException("Error opening session. Cause: " + e, e); } finally { ErrorContext.instance().reset(); } }
主要代碼在 openSessionFromDataSource,首先是從 Configuration 中取出相關的配置,生成 Transaction,接著又創建了一個 Executor,最后返回了 DefaultSqlSession 對象。
這里的 Executor 是什么呢?它其實是一個執行器,SqlSession 的操作會交給 Executor 去執行。MyBatis 的 Executor 常用的有以下幾種:
SimpleExecutor: 默認的 Executor,每個 SQL 執行時都會創建新的 Statement
ResuseExecutor: 相同的 SQL 會復用 Statement
BatchExecutor: 用于批處理的 Executor
CachingExecutor: 可緩存數據的 Executor,用代理模式包裝了其它類型的 Executor
了解了 Executor 的類型后,看一下 configuration.newExecutor(tx, execType, autoCommit) 的代碼:
public Executor newExecutor(Transaction transaction, ExecutorType executorType) { // 默認是 SimpleExecutor executorType = executorType == null ? defaultExecutorType : executorType; executorType = executorType == null ? ExecutorType.SIMPLE : executorType; Executor 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); } // 默認啟動緩存 if (cacheEnabled) { executor = new CachingExecutor(executor); } executor = (Executor) interceptorChain.pluginAll(executor); return executor; }
MyBatis 默認啟用一級緩存,即同一個 SqlSession 會共用同一個緩存,上面代碼最終返回的是 CachingExecutor。
getMapper在創建了 SqlSession 之后,下一步是生成 Mapper 接口的代理類,代碼如下:
publicT getMapper(Class type) { return configuration. getMapper(type, this); }
可以看出是從 configuration 中取得 Mapper,最終調用了 MapperProxyFactory 的 newInstance。MapperProxyFactory 在上一篇文章已經分析過,它是為了給 Mapper 接口生成代理類,其中關鍵的攔截邏輯在 MapperProxy 中,下面是其 invoke 方法:
@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { // 過濾一些不需要被代理的方法 if (Object.class.equals(method.getDeclaringClass())) { return method.invoke(this, args); } else if (isDefaultMethod(method)) { return invokeDefaultMethod(proxy, method, args); } } catch (Throwable t) { throw ExceptionUtil.unwrapThrowable(t); } // 從緩存中獲取 MapperMethod 然后調用 final MapperMethod mapperMethod = cachedMapperMethod(method); return mapperMethod.execute(sqlSession, args); }
MapperProxy 中調用了 MapperMethod 的 execute,下面是部分代碼:
public Object execute(SqlSession sqlSession, Object[] args) { Object result; switch (command.getType()) { case INSERT: { Object param = method.convertArgsToSqlCommandParam(args); result = rowCountResult(sqlSession.insert(command.getName(), param)); break; } case UPDATE: { Object param = method.convertArgsToSqlCommandParam(args); result = rowCountResult(sqlSession.update(command.getName(), param)); break; } case DELETE: { Object param = method.convertArgsToSqlCommandParam(args); result = rowCountResult(sqlSession.delete(command.getName(), param)); break; } case SELECT: if (method.returnsVoid() && method.hasResultHandler()) { executeWithResultHandler(sqlSession, args); result = null; ... }
可以看出,最終調用了 SqlSession 的對應方法,也就是 DefaultSqlSession 中的方法。
select先看一下 DefaultSqlSession 中 select 的代碼:
public void select(String statement, Object parameter, RowBounds rowBounds, ResultHandler handler) { try { MappedStatement ms = configuration.getMappedStatement(statement); executor.query(ms, wrapCollection(parameter), rowBounds, handler); } catch (Exception e) { throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e); } finally { ErrorContext.instance().reset(); } }
select 中調用了 executor 的 query,上面提到,默認的 Executor 是 CachingExecutor,看其中的代碼:
@Override publicList query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException { BoundSql boundSql = ms.getBoundSql(parameterObject); // 獲取緩存的key CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql); return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); } public List query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException { // 獲取緩存 Cache cache = ms.getCache(); if (cache != null) { flushCacheIfRequired(ms); if (ms.isUseCache() && resultHandler == null) { ensureNoOutParams(ms, boundSql); @SuppressWarnings("unchecked") List list = (List ) tcm.getObject(cache, key); if (list == null) { list = delegate. query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); tcm.putObject(cache, key, list); // issue #578 and #116 } return list; } } // 調用代理對象的緩存 return delegate. query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); }
首先檢查緩存中是否有數據,如果沒有再調用代理對象的 query,默認是 SimpleExecutor。Executor 是一個接口,下面有個實現類是 BaseExecutor,其中實現了其它 Executor 通用的一些邏輯,包括 doQuery 以及 doUpdate 等,其中封裝了 JDBC 的相關操作。
updateupdate 的執行與 select 類似, 都是從 CachingExecutor 開始,看代碼:
@Override public int update(MappedStatement ms, Object parameterObject) throws SQLException { // 檢查是否需要刷新緩存 flushCacheIfRequired(ms); // 調用代理類的 update return delegate.update(ms, parameterObject); }
update 會使得緩存的失效,所以第一步是檢查是否需要刷新緩存,接下來再交給代理類去執行真正的數據庫更新操作。
總結本文主要分析了 SqlSession 的執行流程,結合上一篇文章基本了解了 MyBatis 的運行原理。對于 MyBatis 的源碼,還有很多地方沒有深入,例如SQL 解析時參數的處理、一級緩存與二級緩存的處理邏輯等,不過在熟悉 MyBatis 的整體框架之后,這些細節可以在需要用到的時候繼續學習。
如果我的文章對您有幫助,不妨點個贊支持一下(^_^)
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/72760.html
摘要:我認為學習框架源碼分為兩步抓住主線,掌握框架的原理和流程理解了處理思路之后,再去理解面向對象思想和設計模式的用法目前第一步尚有問題,需要多走幾遍源碼,加深下理解,一起加油 這篇文章我們來深入閱讀下Mybatis的源碼,希望以后可以對底層框架不那么畏懼,學習框架設計中好的思想; 架構原理 架構圖 showImg(https://segmentfault.com/img/remote/...
摘要:最終解析出的和依然是設置到中。到這里,初始化部分就結束了。總結的初始化流程主要是解析配置文件,將相關信息保存在中,同時對每個代表的生成代理對象工廠。 簡介 MyBatis 是 Java 開發中非常流行的 ORM 框架,其封裝了 JDBC 并且解決了 Java 對象與輸入參數和結果集的映射,同時又能夠讓用戶方便地手寫 SQL 語句。MyBatis 的行為類似于以下幾行代碼: Class....
摘要:理解與掌握原理分析框架功能架構接口層提供給外部使用的接口,開發人員通過這些本地來操作數據庫。流程分析數據處理過程根據的查找相應的對象。預處理對象,得到對象。傳入和結果處理對象,通過的方法來執行,并對執行結果進行處理。 MyBatis理解與掌握(原理分析) @(MyBatis)[Java, 框架, MyBatis] 功能架構 showImg(https://segmentfault.co...
摘要:目標理清加載解析文件的過程理清執行的過程。先看源碼生成解析配置文件考慮到項目的配置,看下生成和的代碼。在生成的過程中,使用了,這個類繼承了。在該類的構造器中加載了和。下面看一下代碼從緩存中獲取對象執行下面是方法至此,執行完成。 目標: 理清mybatis加載解析mapper文件的過程; 理清mybatis執行SQL的過程。 上一篇文章分析mybatis加載配置的源碼時提到了org....
摘要:一級緩存介紹及相關配置。在這個章節,我們學習如何使用的一級緩存。一級緩存實驗配置完畢后,通過實驗的方式了解一級緩存的效果。源碼分析了解具體的工作流程后,我們隊查詢相關的核心類和一級緩存的源碼進行走讀。 我,后端Java工程師,現在美團點評工作。愛健身,愛技術,也喜歡寫點文字。個人網站: http://kailuncen.me公眾號: KailunTalk (凱倫說) 前言 本文主要涉及...
閱讀 3501·2021-11-24 11:17
閱讀 2289·2021-11-15 11:38
閱讀 3373·2021-10-14 09:42
閱讀 2946·2019-08-30 15:54
閱讀 2032·2019-08-28 18:09
閱讀 546·2019-08-26 11:48
閱讀 1637·2019-08-26 10:48
閱讀 2158·2019-08-26 10:45