最近研究了一下Mybatis,給大家磕叨磕叨,MyBatis框架的核心功能其實不難,無非就是動態(tài)代理和jdbc的操作,難的是寫出來可擴展,高內(nèi)聚,低耦合的規(guī)范的代碼。本文完成的Mybatis功能比較簡單,代碼還有許多需要改進的地方,大家可以結(jié)合Mybatis源碼去動手完善。
一、Mybatis框架流程簡介
在手寫自己的Mybatis框架之前,我們先來了解一下Mybatis,它的源碼中使用了大量的設(shè)計模式,閱讀源碼并觀察設(shè)計模式在其中的應(yīng)用,才能夠更深入的理解源碼(ref:Mybatis源碼解讀-設(shè)計模式總結(jié))。我們對上圖進行分析總結(jié):
mybatis的配置文件有2類
mybatisconfig.xml,配置文件的名稱不是固定的,配置了全局的參數(shù)的配置,全局只能有一個配置文件。
Mapper.xml 配置多個statemement,也就是多個sql,整個mybatis框架中可以有多個Mappe.xml配置文件。
通過mybatis配置文件得到SqlSessionFactory
通過SqlSessionFactory得到SqlSession,用SqlSession就可以操作數(shù)據(jù)了。
SqlSession通過底層的Executor(執(zhí)行器),執(zhí)行器有2類實現(xiàn):
基本實現(xiàn)
帶有緩存功能的實現(xiàn)
MappedStatement是通過Mapper.xml中定義statement生成的對象。
參數(shù)輸入執(zhí)行并輸出結(jié)果集,無需手動判斷參數(shù)類型和參數(shù)下標(biāo)位置,且自動將結(jié)果集映射為Java對象
HashMap,KV格式的數(shù)據(jù)類型
Java的基本數(shù)據(jù)類型
POJO,java的對象
二、梳理自己的Mybatis的設(shè)計思路
根據(jù)上文Mybatis流程,我簡化了下,分為以下步驟:
1.讀取xml文件,建立連接
從圖中可以看出,MyConfiguration負責(zé)與人交互。待讀取xml后,將屬性和連接數(shù)據(jù)庫的操作封裝在MyConfiguration對象中供后面的組件調(diào)用。本文將使用dom4j來讀取xml文件,它具有性能優(yōu)異和非常方便使用的特點。
2.創(chuàng)建SqlSession,搭建Configuration和Executor之間的橋梁
我們經(jīng)常在使用框架時看到Session,Session到底是什么呢?一個Session僅擁有一個對應(yīng)的數(shù)據(jù)庫連接。類似于一個前段請求Request,它可以直接調(diào)用exec(SQL)來執(zhí)行SQL語句。從流程圖中的箭頭可以看出,MySqlSession的成員變量中必須得有MyExecutor和MyConfiguration去集中做調(diào)配,箭頭就像是一種關(guān)聯(lián)關(guān)系。我們自己的MySqlSession將有一個getMapper方法,然后使用動態(tài)代理生成對象后,就可以做數(shù)據(jù)庫的操作了。
3.創(chuàng)建Executor,封裝JDBC操作數(shù)據(jù)庫
Executor是一個執(zhí)行器,負責(zé)SQL語句的生成和查詢緩存(緩存還沒完成)的維護,也就是jdbc的代碼將在這里完成,不過本文只實現(xiàn)了單表,有興趣的同學(xué)可以嘗試完成多表。
4.創(chuàng)建MapperProxy,使用動態(tài)代理生成Mapper對象
我們只是希望對指定的接口生成一個對象,使得執(zhí)行它的時候能運行一句sql罷了,而接口無法直接調(diào)用方法,所以這里使用動態(tài)代理生成對象,在執(zhí)行時還是回到MySqlSession中調(diào)用查詢,最終由MyExecutor做JDBC查詢。這樣設(shè)計是為了單一職責(zé),可擴展性更強。
歡迎大家加入粉絲群:277763288,群內(nèi)免費分享Spring框架、Mybatis框架SpringBoot框架、SpringMVC框架、SpringCloud微服務(wù)、Dubbo框架、Redis緩存、RabbitMq消息、JVM調(diào)優(yōu)、Tomcat容器、MySQL數(shù)據(jù)庫教學(xué)視頻及架構(gòu)學(xué)習(xí)思維導(dǎo)圖
三、實現(xiàn)自己的Mybatis
工程文件及目錄:
首先,新建一個maven項目,在pom.xml中導(dǎo)入以下依賴:

創(chuàng)建我們的數(shù)據(jù)庫xml配置文件:

然后在數(shù)據(jù)庫創(chuàng)建test庫,執(zhí)行如下SQL語句:

創(chuàng)建User實體類,和UserMapper接口和對應(yīng)的xml文件:

基本操作配置完成,接下來我們開始實現(xiàn)MyConfiguration:




用面向?qū)ο蟮乃枷朐O(shè)計讀取xml配置后:

接下來實現(xiàn)我們的MySqlSession,首先的成員變量里得有Excutor和MyConfiguration,代碼的精髓就在getMapper的方法里。

緊接著創(chuàng)建Excutor和實現(xiàn)類:

MyExcutor中封裝了JDBC的操作:


MyMapperProxy代理類完成xml方法和真實方法對應(yīng),執(zhí)行查詢:

到這里,就完成了自己的Mybatis框架,我們測試一下:

歡迎大家加入粉絲群:277763288,群內(nèi)免費分享Spring框架、Mybatis框架SpringBoot框架、SpringMVC框架、SpringCloud微服務(wù)、Dubbo框架、Redis緩存、RabbitMq消息、JVM調(diào)優(yōu)、Tomcat容器、MySQL數(shù)據(jù)庫教學(xué)視頻及架構(gòu)學(xué)習(xí)思維導(dǎo)圖
執(zhí)行結(jié)果:
查詢一個不存在的用戶試試:
到這里我們就大功告成了!
我自己在騰訊課堂上有錄制過一堂手寫mybatis的直播分享,一堂課帶你深入了解mybatis
主要從以下來講述:
1.劃重點啦,花個三五分鐘回顧下MyBatis的要素;
2.想好了才能動手,MyBatis核心流程解密;
3.一路向西,走三步手寫MyBatis,你一定能聽懂!
4.老司機帶你大保健,居然我把MyBatis看懂了;
5.論學(xué)習(xí)源碼的重要性,是時候修煉內(nèi)功了!
6.互聯(lián)網(wǎng)架構(gòu)師技能樹梳理,作為一個互聯(lián)網(wǎng)高級架構(gòu)師你需要掌握的都在這里;
感興趣的可以進群 277763288 獲取這堂課的直播鏈接學(xué)習(xí)學(xué)習(xí),群內(nèi)提供免費的Java架構(gòu)學(xué)習(xí)資料(里面有高可用、高并發(fā)、高性能及分布式、Jvm性能調(diào)優(yōu)、Spring源碼,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多個知識點的架構(gòu)資料)合理利用自己每一分每一秒的時間來學(xué)習(xí)提升自己,不要再用"沒有時間“來掩飾自己思想上的懶惰!趁年輕,使勁拼,給未來的自己一個交代!