透徹理解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

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