<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          MyBatis 的執(zhí)行流程,學(xué)廢了!

          共 9033字,需瀏覽 19分鐘

           ·

          2021-07-20 01:21

          fc4feaf8b29940679f7fe932196e6ba7.webp

          作者:雙子孤狼

          來源:blog.csdn.net/zwx900102/article/details/108455514

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

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

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

          概要

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

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

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

          獲取Mapper接口(getMapper)

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

          1、在調(diào)用getMapper之后,會去Configuration對象中獲取Mapper對象,因為在項目啟動的時候就會把Mapper接口加載并解析存儲到Configuration對象

          d3d4e1af759f8bc387d1311258009ab1.webp

          2、通過Configuration對象中的MapperRegistry對象屬性,繼續(xù)調(diào)用getMapper方法8caad508b1553030fcef94b795852eb2.webp

          3、根據(jù)type類型,從MapperRegistry對象中的knownMappers獲取到當(dāng)前類型對應(yīng)的代理工廠類,然后通過代理工廠類生成對應(yīng)Mapper的代理類4a189b09e2371ed86fa315a7c0909644.webp

          4、最終獲取到我們接口對應(yīng)的代理類MapperProxy對象e0652ef1daa2c31961ce23a62dd451c3.webp而MapperProxy可以看到實現(xiàn)了InvocationHandler,使用的就是JDK動態(tài)代理。853337981c4128654787f38f2d068ae3.webp至此獲取Mapper流程結(jié)束了,那么就有一個問題了MapperRegistry對象內(nèi)的HashMap屬性knownMappers中的數(shù)據(jù)是什么時候存進去的呢?

          Mapper接口和映射文件是何時關(guān)聯(lián)的

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

          1、首先我們會手動調(diào)用SqlSessionFactoryBuilder方法中的build()方法:11e650e866a00906e43566b3d5ae7bea.webp

          2、然后會構(gòu)造一個XMLConfigBuilder對象,并調(diào)用其parse方法:998877e3c55494f0a4cf5b8de310a781.webp

          3、然后會繼續(xù)調(diào)用自己的parseConfiguration來解析配置文件,這里面就會分別去解析全局配置文件的頂級節(jié)點,其他的我們先不看,我們直接看最后解析mappers節(jié)點dc94c6cc9875b13f4656aee149e1e174.webp

          4、繼續(xù)調(diào)用自己的mapperElement來解析mappers文件(這個方法比較長,為了方便截圖完整,所以把字體縮小了1號),可以看到,這里面分了四種方式來解析mappers節(jié)點的配置,對應(yīng)了4種mapper配置方式,而其中紅框內(nèi)的兩種方式是直接配置的xml映射文件,藍框內(nèi)的兩種方式是解析直接配置Mapper接口的方式,從這里也可以說明,不論配置哪種方式,最終MyBatis都會將xml映射文件和Mapper接口進行關(guān)聯(lián)670058352b0efaff119b12ebd15e4482.webp

          5、我們先看第2種和第3中(直接配置xml映射文件的解析方式),會構(gòu)建一個XMLMapperBuilder對象并調(diào)用其parse方法。e0ebd2bda09be0dcd426f95ba70f5b7e.webp但是這里有一個問題,如果有多重繼承或者多重依賴時在這里是可能會無法被完全解析的,比如說三個映射文件互相依賴,那么if里面(假設(shè)是最壞情況)只能加載1個,失敗2個,然后走到下面if之外的代碼又只能加載1個,還有1個會失敗(如下代碼中,只會處理1次,再次失敗并不會繼續(xù)加入incompleteResultMaps):d41c1c0018824403a95c27d455359c42.webp當(dāng)然,這個還是會被解析的,后面執(zhí)行查詢的時候會再次通過不斷遍歷去全部解析完畢,不過有一點需要注意的是,互相引用這種是會導(dǎo)致解析失敗報錯的,所以在開發(fā)過程中我們應(yīng)該避免循環(huán)依賴的產(chǎn)生

          6、解析完映射文件之后,調(diào)用自身方法bindMapperForNamespace,開始綁定Mapper接口和映射文件:4d6a88a40c5bc4846cae052be4a98c4f.webp

          7、調(diào)用Configuration對象的addMapper3003176c34e57dff6100e22dff0e3790.webp

          8、調(diào)用Configuration對象的屬性MapperRegistry內(nèi)的addMapper方法,這個方法就是正式將Mapper接口添加到knownMappers,所以上面getMapper可以直接獲取:a341781c6f1f0296f2bd76a1e190a4c4.webp到這里我們就完成了Mapper接口和xml映射文件的綁定

          9、注意上面紅框里面的代碼,又調(diào)用了一次parse方法,這個parse方法主要是解析注解,比如下面的語句:

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

          所以這個方法里面會去解析@Select等注解,需要注意的是,parse方法里面會同時再解析一次xml映射文件,因為上面我們提到了mappers節(jié)點有4種配置方式,其中兩種配置的是Mapper接口,而配置Mapper接口會直接先調(diào)用addMapper接口,并沒有解析映射文件,所以進入注解解析方法parse之中會需要再嘗試解析一次XML映射文件。f3656dd0794acb8609050ec0cb2ef5a9.webp解析完成之后,還會對Mapper接口中的方法進行解析,并將每個方法的全限定類名作為key存入存入Configuration中的mappedStatements屬性。

          需要指出的是,這里存儲的時候,同一個value會存儲2次,一個全限定名作為key,另一個就是只用方法名(sql語句的id)來作為keyfda60f7336dce8fad9d88915d60698d2.webp所以最終mappedStatements會是下面的情況:63c13604422f460f55def099e8a70b3e.webp事實上如果我們通過接口的方式來編程的話,最后來getStatement的時候,都是根據(jù)全限定名來取的,所以即使有重名對我們也沒有影響,而之所以要這么做的原因其實還是為了兼容早期版本的用法,那就是不通過接口,而是直接通過方法名的方式來進行查詢

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

          這里如果shortName沒有重復(fù)的話,是可以直接通過簡寫來查詢的:

          session.selectList("listAllUser");

          但是通過簡寫來查詢一旦shortName重復(fù)了就會拋出以下異常:672a05e9eda2306e125a118867a1d58e.webp這里的異常其實就是StrickMap的get方法拋出來的:43967b613fc2498b9128cb18ffc67616.webp

          sql執(zhí)行流程分析

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

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

          • 一、尋找sql
          • 二、執(zhí)行sql語句

          尋找sql

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

          1、了解代理模式的應(yīng)該都知道,調(diào)用被代理對象的方法之后實際上執(zhí)行的就是代理對象的invoke方法18139fbf4b8d2c55d20e74f3716d1375.webp

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

          3、接下來,是構(gòu)造一個MapperMethod對象,這個對象封裝了Mapper接口中對應(yīng)的方法信息以及對應(yīng)的sql語句信息:c57880808a14173fbacf6266038054e9.webp這里面就會把要執(zhí)行的sql語句,請求參數(shù),方法返回值全部解析封裝成MapperMethod對象,然后后面就可以開始準(zhǔn)備執(zhí)行sql語句了

          執(zhí)行sql語句

          還是先來看一下執(zhí)行Sql語句的時序圖:6237f644358be6f17aa1ec90cf945dba.webp1、我們繼續(xù)上面的流程進入execute方法:154326badf8f94c318516a6083c714c5.webp

          2、這里面會根據(jù)語句類型以及返回值類型來決定如何執(zhí)行,本人這里返回的是一個集合,故而我們進入executeForMany方法:945d256587f602d622e8c0646845f8d1.webp

          3、這里面首先會將前面存好的參數(shù)進行一次轉(zhuǎn)換,然后繞了這么一圈,回到了起點SqlSession對象,繼續(xù)調(diào)用selectList方法:513bbe6cbf449d3cf608423131f3ddab.webp

          4、接下來又講流程委派給了Execute去執(zhí)行query方法,最終又會去調(diào)用queryFromDatabase方法:2195902cd40cbcaf8f79a4ced85d18a4.webp

          5、到這里之后,終于要進入正題了,一般帶了這種do開頭的方法就是真正做事的,Spring中很多地方也是采用的這種命名方式:600d72eb1a469d7d256e6264ba5082a2.webp注意,前面我們的sql語句還是占位符的方式,并沒有將參數(shù)設(shè)置進去,所以這里在return上面一行調(diào)用prepareStatement方法創(chuàng)建Statement對象的時候會去設(shè)置參數(shù),替換占位符。參數(shù)如何設(shè)置我們先跳過,等把流程執(zhí)行完了我們在單獨分析參數(shù)映射和結(jié)果集映射。

          6、繼續(xù)進入PreparedStatementHandler對象的query方法,可以看到,這一步就是調(diào)用了jdbc操作對象PreparedStatement中的execute方法,最后一步就是轉(zhuǎn)換結(jié)果集然后返回。4ed86772dc1c00c071368288ee559ba5.webp到這里,整個SQL語句執(zhí)行流程分析就結(jié)束了,中途有一些參數(shù)的存儲以及轉(zhuǎn)換并沒有深入進去,因為參數(shù)的轉(zhuǎn)換并不是核心,只要清楚整個數(shù)據(jù)的流轉(zhuǎn)流程,我們自己也可以有自己的實現(xiàn)方式,只要存起來最后我們能重新解析讀出來就行。

          參數(shù)映射

          現(xiàn)在我們來看一下上面在執(zhí)行查詢之前參數(shù)是如何進行設(shè)置的,我們先進入prepareStatement方法:8840b8329f9a1ab8a8b246d81cef151a.webp我們發(fā)現(xiàn),最終是調(diào)用了StatementHandler中的parameterize進行參數(shù)設(shè)置,接下來這里為了節(jié)省篇幅,我們不會一步步點進去,直接進入設(shè)置參數(shù)的方法:ccd5068bb9300ffa18f05bc96fd12a8a.webp上面的BaseTypeHandler是一個抽象類,setNonNullParameter并沒有實現(xiàn),都是交給子類去實現(xiàn),而每一個子類就是對應(yīng)了數(shù)據(jù)庫的一種類型。下圖中就是默認(rèn)的一個子類StringTypeHandler,里面沒什么其他邏輯,就是設(shè)置參數(shù)。7fe74a4ac512ff8fcc766a6b5ba3a810.webp可以看到String里面調(diào)用了jdbc中的setString方法,而如果是int也會調(diào)用setInt方法。看到這些子類如果大家之前閱讀過我前面講的MyBatis參數(shù)配置,應(yīng)該就很明顯可以知道,這些子類就是系統(tǒng)默認(rèn)提供的一些typeHandler。而這些默認(rèn)的typeHandler會默認(rèn)被注冊并和Java對象進行綁定:136cd9a8541dbb4867398d414bec6952.webp正是因為MyBatis中默認(rèn)提供了常用數(shù)據(jù)類型的映射,所以我們寫Sql的時候才可以省略參數(shù)映射關(guān)系,可以直接采用下面的方式,系統(tǒng)可以根據(jù)我們參數(shù)的類型,自動選擇合適的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,所以會優(yōu)先以配置的typeHandler為主不會再去讀取默認(rèn)的映射,如果類型不匹配就會直接報錯了:be667de29be3c51748661d6827758f1a.webp看到這里很多人應(yīng)該就知道了,如果我們自己自定義一個typeHandler,然后就可以配置成我們自己的自定義類。所以接下來就讓我們看看如何自定義一個typeHandler

          自定義typeHandler

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

          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}

          然后執(zhí)行,可以看到,自定義的typeHandler生效了:5e66cf2cc1047904a0df136a2d9efae8.webp

          結(jié)果集映射

          接下來讓我們看看結(jié)果集的映射,回到上面執(zhí)行sql流程的最后一個方法:

          resultSetHandler.handleResultSets(ps)

          結(jié)果集映射里面的邏輯相對來說還是挺復(fù)雜的,因為要考慮到非常多的情況,這里我們就不會去深究每一個細節(jié),直接進入到正式解析結(jié)果集的代碼,下面的5個代碼片段就是一個簡單的但是完整的解析流程:a72deb05e6e7d0f29ba69573918959a2.webpb91b642332fd7bb9b9782bf6f8ebffcd.webpcda31dffe4637f6614943374230cf68d.webp47f89e84f66743fa988375714101c38c.webp86c776420cb89a15e313affd6f166649.webp從上面的代碼片段我們也可以看到,實際上解析結(jié)果集還是很復(fù)雜的,就如我們上一篇介紹的復(fù)雜查詢一樣,一個查詢可以不斷嵌套其他查詢,還有延遲加載等等一些復(fù)雜的特性 的處理,所以邏輯分支是有很多,但是不管怎么處理,最后的核心還是上面的一套流程,最終還是會調(diào)用typeHandler來獲取查詢到的結(jié)果。

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

          自定義typeHandler結(jié)果集

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

          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>?{

          ????/**
          ?????*?設(shè)置參數(shù)
          ?????*/

          ????@Override
          ????public?void?setNonNullParameter(PreparedStatement?preparedStatement,?int?index,?String?param,?JdbcType?jdbcType)?throws?SQLException?{
          ????????System.out.println("設(shè)置參數(shù)->自定義typeHandler生效了");
          ????????preparedStatement.setString(index,param);
          ????}
          ????/**
          ?????*?根據(jù)列名獲取結(jié)果
          ?????*/

          ????@Override
          ????public?String?getNullableResult(ResultSet?resultSet,?String?columnName)?throws?SQLException?{
          ????????System.out.println("根據(jù)columnName獲取結(jié)果->自定義typeHandler生效了");
          ????????return?resultSet.getString(columnName);
          ????}

          ????/**
          ?????*?根據(jù)列的下標(biāo)來獲取結(jié)果
          ?????*/

          ????@Override
          ????public?String?getNullableResult(ResultSet?resultSet,?int?columnIndex)?throws?SQLException?{
          ????????System.out.println("根據(jù)columnIndex獲取結(jié)果->自定義typeHandler生效了");
          ????????return?resultSet.getString(columnIndex);
          ????}

          ????/**
          ?????*?處理存儲過程的結(jié)果集
          ?????*/

          ????@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>

          執(zhí)行之后輸出如下:6eb45430c9c5d0ce255a91e9735ca784.webp因為我們屬性上面只配置了一個屬性,所以只輸出了一次。

          工作流程圖

          上面介紹了代碼的流轉(zhuǎn),可能繞來繞去有點暈,所以我們來畫一個主要的對象之間流程圖來更加清晰的展示一下MyBatis主要工作流程:5b8d7deafaf6eda22a78525cfa6ea282.webp從上面的工作流程圖上我們可以看到,SqlSession下面還有4大對象,這4大對象也很重要,后面學(xué)習(xí)攔截器的時候就是針對這4大對象進行的攔截,關(guān)于這4大對象的具體詳情,我們下一篇文章再展開分析。

          總結(jié)

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

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

          當(dāng)然,其中很多細節(jié)并沒有提到,而看源碼我們也并不需要追求每一行代碼都能看懂,就比如我們一個稍微復(fù)雜一點的業(yè)務(wù)系統(tǒng),即使我們是項目開發(fā)者如果某一個模塊不是本人負責(zé)的,恐怕也很難搞清楚每一行代碼的含義。所以對于MyBatis及其他框架的源碼中也是一樣,首先應(yīng)該從大局入手,掌握整體流程和設(shè)計思想,然后如果對某些實現(xiàn)細節(jié)感興趣,再深入進行了解。

          778b9bc931ed1a3090a8cd881c902b62.webp

          往期推薦

          ce80fcd2b73ac34399e3b5e832638606.webp

          Mybatis中SQL注入攻擊的3種方式,真是防不勝防!


          0877560e4d619da2fc2380ca8d569215.webp

          批處理框架 Spring Batch 這么強,你會用嗎?


          4ea7cd5f58211fc6f8254240661d604c.webp

          面霸篇:MQ 的 5 大關(guān)鍵問題詳解




          瀏覽 30
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  国产视频色情 | 一级黄色AA片 | 青青草成人在线免费观看 | 枕瑶钗十三回兴云弄雨又春风 | 干少妇AV |