MyBatis的SQL執行流程不清楚?看完這一篇就夠了

2020-10-13 追逐仰望星空

推薦學習

前言

MyBatis可能很多人都一直在用,但是MyBatis的SQL執行流程可能並不是所有人都清楚了,那麼既然進來了,通讀本文你將收穫如下:

  • 1、Mapper接口和映射文件是如何進行綁定的
  • 2、MyBatis中SQL語句的執行流程
  • 3、自定義MyBatis中的參數設置處理器typeHandler
  • 4、自定義MyBatis中結果集處理器typeHandler

PS:本文基於MyBatis3.5.5版本源碼

概要

在MyBatis中,利用編程式進行數據查詢,主要就是下面幾行代碼:

SqlSession session = sqlSessionFactory.openSession();UserMapper userMapper = session.getMapper(UserMapper.class);List<LwUser> userList = userMapper.listUserByUserName("孤狼1號");

第一行是獲取一個SqlSession對象,第二行就是獲取UserMapper接口,第三行一行代碼就實現了整個查詢語句的流程,接下來我們就來仔細分析一下第二和第三步。

獲取Mapper接口(getMapper)

第二步是通過SqlSession對象是獲取一個Mapper接口,這個流程還是相對簡單的,下面就是我們調用session.getMapper方法之後的運行時序圖:

  • 1、在調用getMapper之後,會去Configuration對象中獲取Mapper對象,因為在項目啟動的時候就會把Mapper接口加載並解析存儲到Configuration對象

  • 2、通過Configuration對象中的MapperRegistry對象屬性,繼續調用getMapper方法

  • 3、根據type類型,從MapperRegistry對象中的knownMappers獲取到當前類型對應的代理工廠類,然後通過代理工廠類生成對應Mapper的代理類

  • 4、最終獲取到我們接口對應的代理類MapperProxy對象

而MapperProxy可以看到實現了InvocationHandler,使用的就是JDK動態代理。

至此獲取Mapper流程結束了,那麼就有一個問題了MapperRegistry對象內的HashMap屬性knownMappers中的數據是什麼時候存進去的呢?

Mapper接口和映射文件是何時關聯的

Mapper接口及其映射文件是在加載mybatis-config配置文件的時候存儲進去的,下面就是時序圖:

  • 1、首先我們會手動調用SqlSessionFactoryBuilder方法中的build()方法:

  • 2、然後會構造一個XMLConfigBuilder對象,並調用其parse方法:

  • 3、然後會繼續調用自己的parseConfiguration來解析配置文件,這裡面就會分別去解析全局配置文件的頂級節點,其他的我們先不看,我們直接看最後解析mappers節點

  • 4、繼續調用自己的mapperElement來解析mappers文件(這個方法比較長,為了方便截圖完整,所以把字體縮小了1號),可以看到,這裡面分了四種方式來解析mappers節點的配置,對應了4種mapper配置方式,而其中紅框內的兩種方式是直接配置的xml映射文件,藍框內的兩種方式是解析直接配置Mapper接口的方式,從這裡也可以說明,不論配置哪種方式,最終MyBatis都會將xml映射文件和Mapper接口進行關聯

  • 5、我們先看第2種和第3中(直接配置xml映射文件的解析方式),會構建一個XMLMapperBuilder對象並調用其parse方法。

但是這裡有一個問題,如果有多重繼承或者多重依賴時在這裡是可能會無法被完全解析的,比如說三個映射文件互相依賴,那麼if裡面(假設是最壞情況)只能加載1個,失敗2個,然後走到下面if之外的代碼又只能加載1個,還有1個會失敗(如下代碼中,只會處理1次,再次失敗並不會繼續加入incompleteResultMaps):

當然,這個還是會被解析的,後面執行查詢的時候會再次通過不斷遍歷去全部解析完畢,不過有一點需要注意的是,互相引用這種是會導致解析失敗報錯的,所以在開發過程中我們應該避免循環依賴的產生

  • 6、解析完映射文件之後,調用自身方法bindMapperForNamespace,開始綁定Mapper接口和映射文件:

  • 7、調用Configuration對象的addMapper

  • 8、調用Configuration對象的屬性MapperRegistry內的addMapper方法,這個方法就是正式將Mapper接口添加到knownMappers,所以上面getMapper可以直接獲取:

到這裡我們就完成了Mapper接口和xml映射文件的綁定

  • 9、注意上面紅框裡面的代碼,又調用了一次parse方法,這個parse方法主要是解析註解,比如下面的語句:

@Select("select * from lw_user") List<LwUser> listAllUser();

所以這個方法裡面會去解析@Select等註解,需要注意的是,parse方法裡面會同時再解析一次xml映射文件,因為上面我們提到了mappers節點有4種配置方式,其中兩種配置的是Mapper接口,而配置Mapper接口會直接先調用addMapper接口,並沒有解析映射文件,所以進入註解解析方法parse之中會需要再嘗試解析一次XML映射文件。

解析完成之後,還會對Mapper接口中的方法進行解析,並將每個方法的全限定類名作為key存入存入Configuration中的mappedStatements屬性。

需要指出的是,這裡存儲的時候,同一個value會存儲2次,一個全限定名作為key,另一個就是只用方法名(sql語句的id)來作為key

所以最終mappedStatements會是下面的情況:

事實上如果我們通過接口的方式來編程的話,最後來getStatement的時候,都是根據全限定名來取的,所以即使有重名對我們也沒有影響,而之所以要這麼做的原因其實還是為了兼容早期版本的用法,那就是不通過接口,而是直接通過方法名的方式來進行查詢

session.selectList("com.lonelyWolf.mybatis.mapper.UserMapper.listAllUser");

這裡如果shortName沒有重複的話,是可以直接通過簡寫來查詢的:

session.selectList("listAllUser");

但是通過簡寫來查詢一旦shortName重複了就會拋出以下異常:

這裡的異常其實就是StrickMap的get方法拋出來的:

sql執行流程分析

上面我們到了,獲取到的Mapper接口實際上被包裝成為了代理對象,所以我們執行查詢語句肯定是執行的代理對象方法,接下來我們就以Mapper接口的代理對象MapperProxy來分析一下查詢流程。

整個sql執行流程可以分為兩大步驟:

  • 一、尋找sql
  • 二、執行sql語句

尋找sql

首先還是來看一下尋找sql語句的時序圖:

  • 1、了解代理模式的應該都知道,調用被代理對象的方法之後實際上執行的就是代理對象的invoke方法

  • 2、因為我們這裡並沒有調用Object類中的方法,所以肯定走的else。else中會繼續調用MapperProxy內部類MapperMethodInvoker中的方法cachedInvoker,這裡面會有一個判斷,判斷一下我們是不是default方法,因為Jdk1.8中接口中可以新增default方法,而default方法是並不是一個抽象方法,所以也需要特殊處理(剛開始會從緩存裡面取,緩存相關知識我們這裡先不講,後面會單獨寫一篇來分析一下緩存))。

  • 3、接下來,是構造一個MapperMethod對象,這個對象封裝了Mapper接口中對應的方法信息以及對應的sql語句信息:

這裡面就會把要執行的sql語句,請求參數,方法返回值全部解析封裝成MapperMethod對象,然後後面就可以開始準備執行sql語句了

執行sql語句

還是先來看一下執行Sql語句的時序圖:

  • 1、我們繼續上面的流程進入execute方法:

  • 2、這裡面會根據語句類型以及返回值類型來決定如何執行,本人這裡返回的是一個集合,故而我們進入executeForMany方法:

  • 3、這裡面首先會將前面存好的參數進行一次轉換,然後繞了這麼一圈,回到了起點SqlSession對象,繼續調用selectList方法:

  • 3、接下來又講流程委派給了Execute去執行query方法,最終又會去調用queryFromDatabase方法:

  • 4、到這裡之後,終於要進入正題了,一般帶了這種do開頭的方法就是真正做事的,Spring中很多地方也是採用的這種命名方式:

注意,前面我們的sql語句還是佔位符的方式,並沒有將參數設置進去,所以這裡在return上面一行調用prepareStatement方法創建Statement對象的時候會去設置參數,替換佔位符。參數如何設置我們先跳過,等把流程執行完了我們單獨分析參數映射和結果集映射。

  • 5、繼續進入PreparedStatementHandler對象的query方法,可以看到,這一步就是調用了jdbc操作對象PreparedStatement中的execute方法,最後一步就是轉換結果集然後返回。

到這裡,整個SQL語句執行流程分析就結束了,中途有一些參數的存儲以及轉換並沒有深入進去,因為參數的轉換並不是核心,只要清楚整個數據的流轉流程,我們自己也可以有自己的實現方式,只要存起來最後我們能重新解析讀出來就行。

參數映射

現在我們來看一下上面在執行查詢之前參數是如何進行設置的,我們先進入prepareStatement方法:

我們發現,最終是調用了StatementHandler中的parameterize進行參數設置,接下來這裡為了節省篇幅,我們不會一步步點進去,直接進入設置參數的方法:

上面的BaseTypeHandler是一個抽象類,setNonNullParameter並沒有實現,都是交給子類去實現,而每一個子類就是對應了資料庫的一種類型。下圖中就是默認的一個子類StringTypeHandler,裡面沒什麼其他邏輯,就是設置參數。

可以看到String裡面調用了jdbc中的setString方法,而如果是int也會調用setInt方法。
看到這些子類如果大家之前閱讀過我前面講的MyBatis參數配置,應該就很明顯可以知道,這些子類就是系統默認提供的一些typeHandler。而這些默認的typeHandler會默認被註冊並和Java對象進行綁定:

正是因為MyBatis中默認提供了常用數據類型的映射,所以我們寫Sql的時候才可以省略參數映射關係,可以直接採用下面的方式,系統可以根據我們參數的類型,自動選擇合適的typeHander進行映射:

select user_id,user_name from lw_user where user_name=#{userName}

上面這條語句實際上和下面這條是等價的:

select user_id,user_name from lw_user where user_name=#{userName,jdbcType=VARCHAR}

或者說我們可以直接指定typeHandler:

select user_id,user_name from lw_user where user_name=#{userName,jdbcType=VARCHAR,typeHandler=org.apache.ibatis.type.IntegerTypeHandler}

這裡因為我們配置了typeHandler,所以會優先以配置的typeHandler為主不會再去讀取默認的映射,如果類型不匹配就會直接報錯了:

看到這裡很多人應該就知道了,如果我們自己自定義一個typeHandler,然後就可以配置成我們自己的自定義類。
所以接下來就讓我們看看如何自定義一個typeHandler

自定義typeHandler

自定義typeHandler需要實現BaseTypeHandler接口,BaseTypeHandler有4個方法,包括結果集映射,為了節省篇幅,代碼沒有寫上來:

package com.lonelyWolf.mybatis.typeHandler;import org.apache.ibatis.type.BaseTypeHandler;import org.apache.ibatis.type.JdbcType;import java.sql.CallableStatement;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.SQLException;public class MyTypeHandler extends BaseTypeHandler<String> { @Override public void setNonNullParameter(PreparedStatement preparedStatement, int index, String param, JdbcType jdbcType) throws SQLException { System.out.println("自定義typeHandler生效了"); preparedStatement.setString(index,param); }

然後我們改寫一下上面的查詢語句:

select user_id,user_name from lw_user where user_name=#{userName,jdbcType=VARCHAR,typeHandler=com.lonelyWolf.mybatis.typeHandler.MyTypeHandler}

然後執行,可以看到,自定義的typeHandler生效了:

結果集映射

接下來讓我們看看結果集的映射,回到上面執行sql流程的最後一個方法:

resultSetHandler.handleResultSets(ps)

結果集映射裡面的邏輯相對來說還是挺複雜的,因為要考慮到非常多的情況,這裡我們就不會去深究每一個細節,直接進入到正式解析結果集的代碼,下面的5個代碼片段就是一個簡單的但是完整的解析流程:

從上面的代碼片段我們也可以看到,實際上解析結果集還是很複雜的,就如我們上一篇介紹的複雜查詢一樣,一個查詢可以不斷嵌套其他查詢,還有延遲加載等等一些複雜的特性
的處理,所以邏輯分支是有很多,但是不管怎麼處理,最後的核心還是上面的一套流程,最終還是會調用typeHandler來獲取查詢到的結果。

是的,你沒猜錯,這個就是上面我們映射參數的typeHandler,因為typeHandler裡面不只是一個設置參數方法,還有獲取結果集方法(上面設置參數的時候省略了)。

自定義typeHandler結果集

所以說我們還是用上面那個MyTypeHandler 例子來重寫一下取值方法(省略了設置參數方法):

package com.lonelyWolf.mybatis.typeHandler;import org.apache.ibatis.type.BaseTypeHandler;import org.apache.ibatis.type.JdbcType;import java.sql.CallableStatement;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.SQLException;public class MyTypeHandler extends BaseTypeHandler<String> { /** * 設置參數 */ @Override public void setNonNullParameter(PreparedStatement preparedStatement, int index, String param, JdbcType jdbcType) throws SQLException { System.out.println("設置參數->自定義typeHandler生效了"); preparedStatement.setString(index,param); } /** * 根據列名獲取結果 */ @Override public String getNullableResult(ResultSet resultSet, String columnName) throws SQLException { System.out.println("根據columnName獲取結果->自定義typeHandler生效了"); return resultSet.getString(columnName); } /** * 根據列的下標來獲取結果 */ @Override public String getNullableResult(ResultSet resultSet, int columnIndex) throws SQLException { System.out.println("根據columnIndex獲取結果->自定義typeHandler生效了"); return resultSet.getString(columnIndex); } /** * 處理存儲過程的結果集 */ @Override public String getNullableResult(CallableStatement callableStatement, int columnIndex) throws SQLException { return callableStatement.getString(columnIndex); }}

改寫Mapper映射文件配置:

<resultMap id="MyUserResultMap" type="lwUser"> <result column="user_id" property="userId" jdbcType="VARCHAR" typeHandler="com.lonelyWolf.mybatis.typeHandler.MyTypeHandler" /> <result column="user_name" property="userName" jdbcType="VARCHAR" /> </resultMap><select id="listUserByUserName" parameterType="String" resultMap="MyUserResultMap"> select user_id,user_name from lw_user where user_name=#{userName,jdbcType=VARCHAR,typeHandler=com.lonelyWolf.mybatis.typeHandler.MyTypeHandler} </select>

執行之後輸出如下:

因為我們屬性上面只配置了一個屬性,所以只輸出了一次。

工作流程圖

上面介紹了代碼的流轉,可能繞來繞去有點暈,所以我們來畫一個主要的對象之間流程圖來更加清晰的展示一下MyBatis主要工作流程:

從上面的工作流程圖上我們可以看到,SqlSession下面還有4大對象,這4大對象也很重要,後面學習攔截器的時候就是針對這4大對象進行的攔截,關於這4大對象的具體詳情,我們下一篇文章再展開分析。

總結

本文主要分析了MyBatis的SQL執行流程。在分析流程的過程中,我們也舉例論證了如何自定義typeHandler來實現自定義的參數映射和結果集映射,不過MyBatis中提供的默認映射其實可以滿足大部分的需求,如果我們對某些屬性需要特殊處理,那麼就可以採用自定義的typeHandle來實現,相信如果本文如果讀懂了,以下幾點大家應該至少會有一個清晰的認識:

  • 1、Mapper接口和映射文件是如何進行綁定的
  • 2、MyBatis中SQL語句的執行流程
  • 3、自定義MyBatis中的參數設置處理器typeHandler
  • 4、自定義MyBatis中結果集處理器typeHandler

當然,其中很多細節並沒有提到,而看源碼我們也並不需要追求每一行代碼都能看懂,就比如我們一個稍微複雜一點的業務系統,即使我們是項目開發者如果某一個模塊不是本人負責的,恐怕也很難搞清楚每一行代碼的含義。所以對於MyBatis及其他框架的源碼中也是一樣,首先應該從大局入手,掌握整體流程和設計思想,然後如果對某些實現細節感興趣,再深入進行了解。

作者:雙子孤狼

原文連結:https://blog.csdn.net/zwx900102/article/details/108455514

相關焦點

  • Mybatis的SqlSession執行sql過程
    上一篇分析了SqlSession的創建過程,接下來就來到最後一步執行sql的過程了。執行sql總覽首先還是來看下目前分析的代碼所處的位置,具體的代碼如下:之前也說過要使用mybatis操作資料庫,在代碼裡面與mybatis相關的代碼只需要三步:創建SqlSessionFactory對象,利用SqlSessionFactory對象創建SqlSession,最後利用SqlSession執行sql。
  • 圖解MyBatis的SQL執行流程(乾貨)
    整個sql執行流程可以分為兩大步驟:一、尋找sql二、執行sql語句尋找sql首先還是來看一下尋找else中會繼續調用MapperProxy內部類MapperMethodInvoker中的方法cachedInvoker,這裡面會有一個判斷,判斷一下我們是不是default方法,因為Jdk1.8中接口中可以新增default方法,而default方法是並不是一個抽象方法,所以也需要特殊處理(剛開始會從緩存裡面取,緩存相關知識我們這裡先不講,後面會單獨寫一篇來分析一下緩存))。
  • 你真的該進來了解下MyBatis的SQL工作流程了
    整個sql執行流程可以分為兩大步驟:一、尋找sql二、執行sql語句尋找sql首先還是來看一下尋找sql語句的時序圖:else中會繼續調用MapperProxy內部類MapperMethodInvoker中的方法cachedInvoker,這裡面會有一個判斷,判斷一下我們是不是default方法,因為Jdk1.8中接口中可以新增default方法,而default方法是並不是一個抽象方法,所以也需要特殊處理(剛開始會從緩存裡面取,緩存相關知識我們這裡先不講,後面會單獨寫一篇來分析一下緩存))。
  • MyBatis的執行流程詳解
    但是好景不長,最近他就在學習MyBatis這裡遇到了點小麻煩,總是搞不清楚,MyBatis是怎麼一個接口、一個映射文件(寫sql)就可以操作資料庫了呢?它的執行流程到底是怎麼樣的呢?帶著這樣的疑問,猿小二開始了他的探索....
  • MyBatis的執行流程詳解,值得收藏
    但是好景不長,最近他就在學習MyBatis這裡遇到了點小麻煩,總是搞不清楚,MyBatis是怎麼一個接口、一個映射文件(寫sql)就可以操作資料庫了呢?它的執行流程到底是怎麼樣的呢?帶著這樣的疑問,猿小二開始了他的探索....
  • SpringBoot整合MyBatis+詳細列印執行SQL語句
    圖片來源網路為啥寫這篇文章呢,有人可能會說springBoot的mybatis的starter都有了寫這篇文章不是多此一舉嗎其實我一開始也是使用mybatis的starter的,但是當我需要配置一個讓mybatis可以將執行sql語句列印出來的配置時,難到了我,所以才想和大家分享一下springBoot整合MyBatis的方法,如果大家有更好地方法,歡迎討論。
  • Mybatis工作流程及其原理
    3.SqlSession (會話): 一個既可以發送 SQL 執行返回結果,也可以獲取Mapper的接口。4.SQL Mapper (映射器): 它由一個Java接口和XML文件(或註解)構成,需要給出對應的SQL和映射規則,它負責發送SQL去執行,並返回結果。
  • Mybatis的sql組裝詳解
    上一篇分析了SqlSession執行sql的過程,其中並沒有分析sql是從哪裡來的,今天就來仔細分析下。Sql來源從上一篇的最後一步執行sql那裡倒推sql的來源,源碼主要過程如下圖:可以看到最後是通過BoundSql直接獲取的sql,然後往前倒推最後發現是通過MappedStatement的getBoundSql方法返回的。
  • mybatis的Configuration詳解
    上一篇介紹了mybatis中SqlSessionFactory的創建過程,今天來學習它默認實現中的唯一屬性Configuration。利用mybatis查出數據只用三步:創建SqlSessionFactory、通過SqlSessionFactory創建SqlSession、SqlSession執行selectOne方法。
  • Spring MyBatis初始化解析+sql執行流程
    當執行數據操作時,會調用代理類的invoke方法,然後查詢數據,並返回相應的結果。在Spring容器下 mybatis一級緩存是沒有效果的 因為每次執行完 sqlSession都會被關閉private class SqlSessionInterceptor implements
  • MyBatis高級應用之逆向工程自動生成SQL語句
    MyBatis的一個主要的特點就是需要程式設計師自己編寫sql,那麼如果表太多的話,難免會很麻煩,所以mybatis官方提供了一個逆向工程,可以針對單表自動生成mybatis執行所需要的代碼,一般在開發中,常用的逆向工程方式是通過資料庫的表生成代碼。下圖為逆向工程依賴的jar包。
  • Mybatis中SqlSource解析流程詳解
    在通過註解實現mapper的流程中是在MapperAnnotationBuilder類的parseStatement方法中對SqlSource進行初始化,初始化代碼如下圖:通過xml文件實現mapper的流程中是在XMLStatementBuilder類的parseStatementNode方法中對SqlSource進行初始化
  • 從 JDBC 到 Mybatis,看這篇就夠了
    由於 ORM 可以自動對 Entity 對象與資料庫中的 Table 進行欄位與屬性的映射,所以我們實際可能已經不需要一個專用的、龐大的數據訪問層。ORM 提供了對資料庫的映射,不用 sql 直接編碼,能夠像操作對象一樣從資料庫獲取數據。缺點:犧牲程序的執行效率和會固定思維模式,降低了開發的靈活性。
  • Java程式設計師開發必備 MyBatis高級應用之逆向工程自動生成SQL語句
    MyBatis的一個主要的特點就是需要程式設計師自己編寫sql,那麼如果表太多的話,難免會很麻煩,所以mybatis官方提供了一個逆向工程,可以針對單表自動生成mybatis執行所需要的代碼,一般在開發中,常用的逆向工程方式是通過資料庫的表生成代碼。
  • Mybatis中mapper的xml解析詳解
    的流程,關鍵代碼如下:解析mapper有4種情況可以分成根據類解析和根據xml文件解析兩類,這次是後面這類,主體方法還是比較簡單。初始化過程源碼已在上圖的右側部分,初始化XMLMapperBuilder方法的流程與加載mybatis的配置文件關鍵類XMLConfigBuilder及其相似,都是通過xml文件創建XPathParser對象作為XMLMapperBuilder對象的parse屬性。
  • 如何回懟面試官的MyBatis執行過程初始化執行問題
    SQL執行過程而且在面試會問到一下關於MyBatis初始化的問題,比如:Mybatis需要初始化哪些?(詳細內容請看下個小節),使用GenericTokenParser::parser()負責將 SQL 語句中的 #{} 替換成相應的 ?
  • 面試:談談你對MyBatis執行過程之SQL執行過程理解
    這樣我們對於Mybatis的整個執行流程都熟悉了,在開發遇到問題也可以很快定位到問題。更重要的,在面試中遇到面試官諮詢Mybatis的知識點的時候,可以很順暢的把這一套流程講出來,面試也會覺得你已掌握Mybatis知識點了。
  • 使用mybatis-plus必須掌握的基本技能,自定義SQL和分頁查詢
    1 創建項目先創建一個集成了mybatis-plus的spring boot項目。參考小編的另一篇文章【2 自定義SQLMybatis-plus自帶的條件構造器雖然很強大,在項目實戰的時候,各種複雜的應用場景,你肯定避免不了要寫稍微複雜一點的sql語句,又或者一些朋友習慣通過寫sql來操作數據,那麼小編來說說mybatis-puls是怎麼自定義sql語句。
  • 最強解讀MyBatis是如何執行SQL語句的?
    本文內容較多,要耐心看完了,文末為大家準了彩蛋--最權威、最詳細的mybatis教程,別忘了購買學習!目錄MyBatis 如何獲取資料庫源?MyBatis 如何獲取 sql 語句?MyBatis 如何執行 sql 語句?MyBatis 如何實現不同類型數據之間的轉換?在過去程式設計師使用JDBC連接資料庫,總會帶來諸多不便。
  • mybatis如何防止SQL注入?
    sql注入發生的時間,sql注入發生的階段在sql預編譯階段,當編譯完成的sql不會產生sql注入一、採用jdbc操作數據時候String sql = &34;+id; PreparedStatement prepareStatement