以下文章來源于公眾號(hào)“程序員求職之路”
在java程序員的世界里笙瑟,最熟悉的開源軟件除了 Spring,Tomcat癞志,還有誰呢往枷?當(dāng)然是 Mybatis 了。MyBatis 是一個(gè)被廣泛應(yīng)用的持久化框架凄杯。代理模式可以認(rèn)為是Mybatis的核心使用的模式错洁。本文通過代理模式,帶你掌握MyBatis原理戒突。
代理模式含義是為其他的對(duì)象提供一種代理以難以控制對(duì)這個(gè)對(duì)象的訪問屯碴,簡而言之即如果有一些我們不能做不想做的事情,可以委托別人去做膊存。優(yōu)點(diǎn)是可以不用修改源代碼导而。
代理模式的角色分類有抽象主題(接口)、具體主題(真實(shí)類)和proxy(代理類)隔崎。
代理模式又分為靜態(tài)代理和動(dòng)態(tài)代理今艺,動(dòng)態(tài)代理又有兩種,一種是JDK動(dòng)態(tài)代理,一種是Cglib爵卒。
靜態(tài)代理是必須要有java源文件虚缎,通過java編譯器轉(zhuǎn)換為.calss文件,通過轉(zhuǎn)換轉(zhuǎn)為byte類加載器再進(jìn)行加載钓株。
動(dòng)態(tài)代理直接在運(yùn)行時(shí)就生成byte文件直接通過類加載器進(jìn)行加載实牡。
JDK動(dòng)態(tài)代理要求必須是實(shí)現(xiàn)接口那種方式,否則不能進(jìn)行動(dòng)態(tài)代理轴合。
Cglib支持不是接口的類能進(jìn)行動(dòng)態(tài)代理创坞,但是Cglib不能代理被final修飾的方法。Springboot 2.x版本spring已經(jīng)添加了cglib受葛。
如果是實(shí)現(xiàn)接口類型的就用jdk動(dòng)態(tài)代理题涨,如果不是接口類型就用Cglib豪椿。
原因是代碼運(yùn)行的時(shí)自動(dòng)生成代理類,幫我們做額外我們想做的事情携栋。
動(dòng)態(tài)代理實(shí)現(xiàn)方法有JDK Proxy和用JDK里的工具自動(dòng)生成動(dòng)態(tài)代理類兩種搭盾。
Client假如就是main方法,doSomething就是調(diào)用任何方法它都會(huì)生成$Proxy類婉支,$Proxy類調(diào)用invoke方法鸯隅,走到我們自己定義的任何代理類方法調(diào)用invoke開始做增強(qiáng)代碼邏輯的操作,然后再調(diào)用被代理對(duì)象的方法向挖。
1首先對(duì)我們傳進(jìn)來的接口InvocationHandler全部克隆了一遍(必須實(shí)現(xiàn)Invocationhandler的接口)
2生成一個(gè)代理類class對(duì)象(get proxy0生成對(duì)象)
3根據(jù)代理類對(duì)象cl蝌以,獲取構(gòu)造器
4根據(jù)構(gòu)造器和InvocationHandler實(shí)現(xiàn)類,創(chuàng)建代理類實(shí)例(使用newInstance)
mapper語句
這是一段Mybatis的一個(gè)查詢數(shù)據(jù)庫數(shù)據(jù)的代碼何之,通過mapper找到配置文件的sql語句,執(zhí)行sql語句獲取數(shù)據(jù)跟畅,但是mapper.selectBlogById()它是沒有實(shí)現(xiàn)類的,那是怎么實(shí)現(xiàn)操作方法的呢溶推?
打印一下類信息徊件,發(fā)現(xiàn)此時(shí)的mapper其實(shí)是動(dòng)態(tài)代理出來的類了
發(fā)現(xiàn)輸出的是jdk的動(dòng)態(tài)代理
那MyBatis中的hadler是什么呢,是MapperProxy里的invoke方法蒜危,也就是說能夠不實(shí)現(xiàn)接口就直接調(diào)用來運(yùn)行sql語句是通過了jdk動(dòng)態(tài)代理實(shí)現(xiàn)的虱痕,sql操作都在mapperMethod里的execute里實(shí)現(xiàn)的。
Mybatis插件
Mybatis攔截器辐赞、分頁插件等一些其他插件都是使用了動(dòng)態(tài)代理部翘,那么mybats里有專門的Plugin代理類, 下圖二Plugin類的invoke
Mybatis的連接池
如果不指定spring管理數(shù)據(jù)庫連接响委,mybatis也是有連接池的新思,而池的操作mybatis也是采用動(dòng)態(tài)代理。因?yàn)閏onnection的連接自己是不可能把自己還給池子里的赘风,而connection本身沒有池夹囚,那么只能通過代理方式增強(qiáng)連接池的功能,代理類幫它把connection放回池操作等等贝次。
invoke的實(shí)現(xiàn)就是假如你要釋放連接崔兴,那么判斷方法是不是CLOSE彰导,然后把連接放入dataSource容器蛔翅,最后return也都是調(diào)回被代理方法的本身操作的方法
Mybatis的日志
Mybatis會(huì)打印執(zhí)行sql日記,肯定不能在業(yè)務(wù)里寫位谋,所以也需要代理模式
ConnectionLogger打印日志山析。
代理模式在MyBatis中經(jīng)常使用,希望這篇文章能讓你掌握MyBatis原理掏父,進(jìn)一步提升操作數(shù)據(jù)庫能力笋轨。