前言
MyBatis出嘹,曾經(jīng)給我的感覺(jué)是一個(gè)很神奇的東西焚辅,我們只需要按照規(guī)范寫好XXXMapper.xml以及XXXMapper.java接口。要知道我們并沒(méi)有提供XXXMapper.java的實(shí)現(xiàn)類,MyBatis到底是怎么做到這一點(diǎn)的呢慎陵?有人會(huì)說(shuō)是動(dòng)態(tài)代理仲闽,現(xiàn)在我就來(lái)通過(guò)手寫一個(gè)迷你版的MyBatis來(lái)徹底理解它的設(shè)計(jì)思想脑溢!
如果你不了解JDK動(dòng)態(tài)代理,請(qǐng)參考:《純手寫實(shí)現(xiàn)JDK動(dòng)態(tài)代理》
如果你不了解MyBatis的使用赖欣,請(qǐng)參考:《MyBatis+Spring MVC開(kāi)發(fā)指南(一)》
如果你對(duì)Spring MVC感興趣屑彻,請(qǐng)參考:《寫出我的第一個(gè)框架:迷你版Spring MVC》
動(dòng)手寫一個(gè)迷你版的MyBatis
其實(shí)對(duì)于MyBatis最為關(guān)鍵的就在于:
XXXMapper mapper = sqlSession.getMapper(XXXMapper.class);
大家可以以這個(gè)為切入口,進(jìn)行源碼跟蹤顶吮,容易得到上面的調(diào)用鏈社牲。
我們先來(lái)看一下迷你版MyBatis的整體框架思路:
執(zhí)行器MyExecutor:
在MyBatis中,比如說(shuō)select有多種形式悴了,比如selectOne,selectList搏恤,那么其實(shí)到最后,還是向JDBC發(fā)出一個(gè)SQL而已湃交。對(duì)于執(zhí)行器而言熟空,其實(shí)對(duì)于查詢,提供一個(gè)query接口就可以了巡揍。
這里痛阻,為了簡(jiǎn)便,直接執(zhí)行已經(jīng)處理好的SQL語(yǔ)句(動(dòng)態(tài)SQL以及輸入類型腮敌,這不是迷你版MyBatis關(guān)心的)阱当。另外執(zhí)行器的實(shí)現(xiàn)類MyBaseExecutor其實(shí)就是一段JDBC的操作代碼。
這里為了簡(jiǎn)化處理糜工,在RequestMapping這塊硬編碼了弊添。
StudentMapper.java/StudentMapper.xml:
這里,為了不牽涉到XML的解析過(guò)程捌木,直接提供已經(jīng)處理完畢的結(jié)果油坝。其實(shí)就是namespace/statementID/SQL的存儲(chǔ)、映射。
對(duì)外暴露的API接口(MySqlSession):
從這里澈圈,你能夠看到一些端倪:
第一彬檀,MyDefaultSqlSession持有執(zhí)行器的引用,調(diào)用selectOne等方法瞬女,就是在調(diào)用執(zhí)行器的query方法窍帝。
第二,在getMapper的獲取過(guò)程中诽偷,我們看到了具體業(yè)務(wù)處理Handler的身影:MyMapperProxy坤学,根據(jù)JDK動(dòng)態(tài)代理的知識(shí),我們知道报慕,最終都是要回調(diào)Handler的invoke方法完成的深浮。
MyMapperProxy:
當(dāng)invoke方法被調(diào)用時(shí),我們根據(jù)調(diào)用的方法眠冈,進(jìn)行反射飞苇,得到namespace以及對(duì)應(yīng)的SQL,然后洋闽,我們把SQL交給sqlSession進(jìn)行執(zhí)行即可玄柠。
啟動(dòng)測(cè)試類Bootstrap:
看到?jīng)]有,我們完全通過(guò)自己的類诫舅,自己的理解羽利,去實(shí)現(xiàn)了和MyBatis一樣的功能!
OK刊懈,一個(gè)迷你版的MyBatis就竣工了这弧,有一種油然而生的成就感,哈哈~