本篇文章給大家分享的是有關(guān)如何深度剖析Java反射機(jī)制中的動(dòng)態(tài)代理原理,小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話(huà)不多說(shuō),跟著小編一起來(lái)看看吧。
創(chuàng)新互聯(lián)建站服務(wù)項(xiàng)目包括貢井網(wǎng)站建設(shè)、貢井網(wǎng)站制作、貢井網(wǎng)頁(yè)制作以及貢井網(wǎng)絡(luò)營(yíng)銷(xiāo)策劃等。多年來(lái),我們專(zhuān)注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,貢井網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶(hù)以成都為中心已經(jīng)輻射到貢井省份的部分城市,未來(lái)相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶(hù)的支持與信任!
動(dòng)態(tài)代理類(lèi)原理 示例代碼參見(jiàn)《Java反射機(jī)制剖析:簡(jiǎn)單談?wù)剟?dòng)態(tài)代理》)
a) 理解上面的動(dòng)態(tài)代理示例流程
b) 代理接口實(shí)現(xiàn)類(lèi)源代碼剖析
咱們一起來(lái)剖析一下代理實(shí)現(xiàn)類(lèi)($Proxy0)的源代碼和整個(gè)動(dòng)態(tài)代理的流程。
$Proxy0生成的代碼如下:
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.lang.reflect.UndeclaredThrowableException; public final class $Proxy0 extends Proxy implements Manager { private static Method m1; private static Method m0; private static Method m3; private static Method m2; static { try { m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") }); m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]); m3 = Class.forName("com.ml.test.Manager").getMethod("test", new Class[0]); m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]); } catch (NoSuchMethodException nosuchmethodexception) { throw new NoSuchMethodError(nosuchmethodexception.getMessage()); } catch (ClassNotFoundException classnotfoundexception) { throw new NoClassDefFoundError(classnotfoundexception.getMessage()); } } public $Proxy0(InvocationHandler invocationhandler) { super(invocationhandler); } @Override public final boolean equals(Object obj) { try { return ((Boolean) super.h.invoke(this, m1, new Object[] { obj })) .booleanValue(); } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } @Override public final int hashCode() { try { return ((Integer) super.h.invoke(this, m0, null)).intValue(); } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } public final void test() { try { super.h.invoke(this, m3, null); return; } catch (Error e) { } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } @Override public final String toString() { try { return (String) super.h.invoke(this, m2, null); } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } }
引入眼簾的是這個(gè)代理接口實(shí)現(xiàn)類(lèi)實(shí)現(xiàn)了業(yè)務(wù)類(lèi)的接口(也就是例子中的UserManager接口),又繼承了基類(lèi)Proxy類(lèi);
接著就是構(gòu)造函數(shù),在構(gòu)造方法中把BusinessHandler傳過(guò)去,接著$Proxy0調(diào)用父類(lèi)Proxy的構(gòu)造器,為h賦值(這里要看Proxy的構(gòu)造方法);
隨后看到的就是這個(gè)類(lèi)重寫(xiě)了Proxy類(lèi)的Equals、hashCode、toString方法,又實(shí)現(xiàn)了業(yè)務(wù)類(lèi)接口的方法(即UserManager的test方法),具體重寫(xiě)和實(shí)現(xiàn)都是用到的super.h.invoke(即Proxy.h.invoke)這個(gè)方法。
簡(jiǎn)單分析完這個(gè)代理接口實(shí)現(xiàn)類(lèi),咱們下面來(lái)整體看一下這個(gè)動(dòng)態(tài)代理是怎么實(shí)現(xiàn)的:
首先客戶(hù)端初始化了BusinessHandler類(lèi),調(diào)用這個(gè)類(lèi)的newProxyInstance(new UserManagerImpl())方法來(lái)初始化了上面的代理接口實(shí)現(xiàn)類(lèi);
接下來(lái)代理接口實(shí)現(xiàn)類(lèi)通過(guò)構(gòu)造函數(shù)把BusinessHandler傳過(guò)去(也就是代碼中的this),并通過(guò)Proxy的構(gòu)造函數(shù)給h賦值;
隨后再客戶(hù)端就能實(shí)例化出代理接口實(shí)現(xiàn)類(lèi)$Proxy0,我們把它強(qiáng)制轉(zhuǎn)換為業(yè)務(wù)實(shí)現(xiàn)接口(UserManager)類(lèi)型的(為什么要強(qiáng)制轉(zhuǎn)換,這里非常有意思,如果不強(qiáng)制轉(zhuǎn)換就會(huì)報(bào)錯(cuò),這里很好解釋?zhuān)驗(yàn)楫?dāng)前的環(huán)境根本不會(huì)知道這個(gè)代理接口實(shí)現(xiàn)類(lèi)$Proxy0既繼承Proxy又實(shí)現(xiàn)業(yè)務(wù)實(shí)現(xiàn)接口UserManager,但是強(qiáng)制轉(zhuǎn)換成UserManager它是可以做到的,因?yàn)楫?dāng)前環(huán)境中就有UserManager。這就是反射的厲害之處,可以在運(yùn)行時(shí)動(dòng)態(tài)調(diào)用任何一個(gè)類(lèi)并可以使用這個(gè)類(lèi)的具體細(xì)節(jié)。);
之后當(dāng)我們調(diào)用test方法的時(shí)候其實(shí)是調(diào)用了$Proxy0中的test方法,這個(gè)方法的實(shí)現(xiàn)是通過(guò)Proxy.h的invoke方法實(shí)現(xiàn)的(即調(diào)用了BusinessHandler.invoke方法);
之后在調(diào)用了Method的invoke方法(這時(shí)的參數(shù)是this,和args)。
這樣就調(diào)用了UserManagerImpl的對(duì)應(yīng)方法,之后返回給客戶(hù)端。
到此就完成了整個(gè)的調(diào)用關(guān)系。
反射,反射,程序員的快樂(lè)
通過(guò)上篇文章對(duì)動(dòng)態(tài)代理進(jìn)行了深度剖析,現(xiàn)在想起來(lái)還感覺(jué)非常有意思,這里面其實(shí)最根本的機(jī)制就是反射機(jī)制,運(yùn)行時(shí)動(dòng)態(tài)實(shí)例化任何一個(gè)類(lèi),并且調(diào)用它的具體細(xì)節(jié)?,F(xiàn)在反看動(dòng)態(tài)代理的示例,其實(shí)發(fā)現(xiàn)這里最關(guān)鍵的還是在就在Proxy.newProxyInstance(..)方法執(zhí)行時(shí)生成了$Proxy0的內(nèi)存字節(jié)碼這一點(diǎn)上,當(dāng)我們有了內(nèi)存字節(jié)碼,我們的反射就會(huì)大顯威力,這樣才有了我們之后的一系列的調(diào)用關(guān)系。
通過(guò)反射機(jī)制的分析和動(dòng)態(tài)代理示例的剖析,發(fā)現(xiàn)編程是一件多么有意思的事情,以至于我們沉浸其中不能自拔。
以上就是如何深度剖析Java反射機(jī)制中的動(dòng)態(tài)代理原理,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見(jiàn)到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。