之前講了《零基礎(chǔ)帶你看Spring源碼——IOC控制反轉(zhuǎn)》,本來打算下一篇講講Srping的AOP的碉哑,但是其中會涉及到Java的動態(tài)代理挚币,所以先單獨一篇來了解下Java的動態(tài)代理到底是什么,Java是怎么實現(xiàn)它的扣典。
動態(tài)代理看起來好像是個什么高大上的名詞妆毕,但其實并沒有那么復(fù)雜,直接從字面就很容易理解贮尖。動態(tài)地代理笛粘,可以猜測一下它的含義,在運(yùn)行時動態(tài)地對某些東西代理,代理它做了其他事情薪前。先不去搞清楚這個動態(tài)代理真正的含義润努,我們來舉個生動的例子來理解下它到底做了什么。
一個例子
一個程序員Developer序六,他會開發(fā)code任连,他調(diào)試debug。
程序員有很多分類例诀,其中有Java程序員JavaDeveloper随抠,他會開發(fā)Java代碼,會調(diào)試Java代碼繁涂。
但是呢拱她,有個叫Zack的程序員它在開發(fā)之前,會祈禱一下扔罪,這樣他開發(fā)的代碼就不會有bug秉沼。
Zack的這種“特異功能”是后天練出來的,并沒有哪種程序員有這種特性矿酵。雖然我們也可以定義一個擁有這樣特性的程序員唬复,但是擁有各種亂七八糟特性的程序千千萬。我們什么時候才能定義完全肮,而能保證不漏呢敞咧?
其實我們沒有必要去定義他,因為他是后天養(yǎng)成的辜腺,我們應(yīng)該在這個程序員的成長期去實現(xiàn)這個特性休建,而不是在他出生之前定義。
我們來看下代碼是怎么實現(xiàn)的
如果Zack只是一個普通的Java程序員评疗,那么他的開發(fā)結(jié)果是
Zack is coding java
Zack is debugging java
但是真正的Zack(代理后)
Zack is praying for the code!
Zack is coding java
Zack's have no bug测砂!No need to debug!
Proxy.newProxyInstance()
回看下上面是如何使用動態(tài)代理的使用。生成一個實例對象百匆,然后用Proxy的newInstance方法對這個實例對象代理生成一個代理對象砌些。
這里有一個非常關(guān)鍵的人,也是比較少人去理解它的加匈。為什么要傳zack的類加載和zack的接口呢寄症?
有沒有留意到zackProxy的類型是Developer,而不是一個通過implements關(guān)鍵詞實現(xiàn)的類矩动。因為zack在被代理后生成的對象,是通過Developer接口的字節(jié)碼增強(qiáng)方式創(chuàng)建的類而構(gòu)造出來的释漆。它是一個臨時構(gòu)造的實現(xiàn)類的對象悲没。
看下newProxyInstance()的接口定義
這三個參數(shù)具體的含義來看看注解是怎么描述的
- loder,選用的類加載器。因為代理的是zack示姿,所以一般都會用加載zack的類加載器甜橱。
- interfaces,被代理的類所實現(xiàn)的接口栈戳,這個接口可以是多個岂傲。
- h,綁定代理類的一個方法子檀。
loder和interfaces基本就是決定了這個類到底是個怎么樣的類镊掖。而h是InvocationHandler,決定了這個代理類到底是多了什么功能褂痰。所以動態(tài)代理的內(nèi)容重點就是這個InvocationHandler亩进。
InvocationHandler
根據(jù)注解描述可知,InvocationHandler作用就是缩歪,當(dāng)代理對象的原本方法被調(diào)用的時候归薛,會綁定執(zhí)行一個方法,這個方法就是InvocationHandler里面定義的內(nèi)容匪蝙,同時會替代原本方法的結(jié)果返回主籍。
InvocationHandler接收三個參數(shù)
- proxy,代理后的實例對象逛球。
- method千元,對象被調(diào)用方法。
- args需忿,調(diào)用時的參數(shù)诅炉。
在上面的例子里,
如果最后的return語句改成
return method.invoke(proxy, agrs);
invoke的對象不是zack屋厘,而是proxy涕烧,根據(jù)上面的說明猜猜會發(fā)生什么?
是的汗洒,會不停地循環(huán)調(diào)用议纯。因為proxy是代理類的對象,當(dāng)該對象方法被調(diào)用的時候溢谤,會觸發(fā)InvocationHandler瞻凤,而InvocationHandler里面又調(diào)用一次proxy里面的對象,所以會不停地循環(huán)調(diào)用世杀。并且阀参,proxy對應(yīng)的方法是沒有實現(xiàn)的桥胞。所以是會循環(huán)的不停報錯
動態(tài)代理的原理
通過上面的講解蔗包,相信大家對動態(tài)代理的使用理解得比較深刻了。那動態(tài)代理到底是怎么實現(xiàn)的呢符欠,我們來看看源碼其中關(guān)鍵的地方。
在newProxyInstance()發(fā)放中有這樣幾段衙荐。
其實大概就是把接口復(fù)制出來捞挥,通過這些接口和類加載器,拿到這個代理類cl忧吟。然后通過反射的技術(shù)復(fù)制拿到代理類的構(gòu)造函數(shù)(這部分代碼在Class類中的getConstructor0方法)砌函,最后通過這個構(gòu)造函數(shù)new個一對象出來,同時用InvocationHandler綁定這個對象溜族。
動態(tài)代理的使用場景
動態(tài)代理的好處我們從例子就能看出來讹俊,它比較靈活,可以在運(yùn)行的時候才切入改變類的方法斩祭,而不需要預(yù)先定義它劣像。
動態(tài)代理一般我們比較少去手寫,但我們用得其實非常多摧玫。在Spring項目中用的注解耳奕,例如依賴注入的@Bean、@Autowired诬像,事務(wù)注解@Transactional等都有用到屋群,換言之就是Srping的AOP(切面編程)。
這種場景的使用是動態(tài)代理最佳的落地點坏挠,可以非常靈活地在某個類芍躏,某個方法,某個代碼點上切入我們想要的內(nèi)容降狠,就是動態(tài)代理其中的內(nèi)容对竣。所以下一篇我們來細(xì)致了解下Spring的AOP到底是怎么使用動態(tài)代理的。
如果覺得還不錯榜配,請關(guān)注微信公眾號:Zack說碼