透徹理解MyBatis設計思想之手寫實現

MyBatis,曾經給我的感覺是一個很神奇的東西,我們只需要按照規範寫好XXXMapper.xml以及XXXMapper.java接口。要知道我們並沒有提供XXXMapper.java的實現類,MyBatis到底是怎麼做到這一點的呢?有人會說是動態代理,現在我就來通過手寫一個迷你版的MyBatis來徹底理解它的設計思想!

動手寫一個迷你版的MyBatis

MyBatis原理架構圖

其實對於MyBatis最爲關鍵的就在於:

XXXMapper mapper = sqlSession.getMapper(XXXMapper.class);

大家可以以這個爲切入口,進行源碼跟蹤,容易得到上面的調用鏈。

我們先來看一下迷你版MyBatis的整體框架思路:

迷你版MyBatis

執行器MyExecutor:

MyExecutor提供query方法

在MyBatis中,比如說select有多種形式,比如selectOne,selectList,那麼其實到最後,還是向JDBC發出一個SQL而已。對於執行器而言,其實對於查詢,提供一個query接口就可以了。

這裏,爲了簡便,直接執行已經處理好的SQL語句(動態SQL以及輸入類型,這不是迷你版MyBatis關心的)。另外執行器的實現類MyBaseExecutor其實就是一段JDBC的操作代碼。

query的JDBC實現

這裏爲了簡化處理,在RequestMapping這塊硬編碼了。

StudentMapper.java/StudentMapper.xml:

Mapper接口

Mapper.xml

這裏,爲了不牽涉到XML的解析過程,直接提供已經處理完畢的結果。其實就是namespace/statementID/SQL的存儲、映射。

對外暴露的API接口(MySqlSession):

MySqlSession

MySqlSession實現

從這裏,你能夠看到一些端倪:

第一,MyDefaultSqlSession持有執行器的引用,調用selectOne等方法,就是在調用執行器的query方法。

第二,在getMapper的獲取過程中,我們看到了具體業務處理Handler的身影:MyMapperProxy,根據JDK動態代理的知識,我們知道,最終都是要回調Handler的invoke方法完成的。

MyMapperProxy:

MyMapperProxy

當invoke方法被調用時,我們根據調用的方法,進行反射,得到namespace以及對應的SQL,然後,我們把SQL交給sqlSession進行執行即可。

啓動測試類Bootstrap:

Bootstrap

看到沒有,我們完全通過自己的類,自己的理解,去實現了和MyBatis一樣的功能!

OK,一個迷你版的MyBatis就竣工了,有一種油然而生的成就感,哈哈~

作者:張豐哲 鏈接:https://www.jianshu.com/p/73ee8caddc68

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章