Java動(dòng)態(tài)代理語法Proxy類原理詳解
1、前言
寫動(dòng)態(tài)代理的代碼涉及了一個(gè)非常重要的類 Proxy,通過Proxy的靜態(tài)方法newProxyInstance才會(huì)動(dòng)態(tài)創(chuàng)建代理對(duì)象。
2、newProxyInstance方法
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
三個(gè)參數(shù)分別表示: loader表示類加載器, interfaces表示代碼要用來代理的接口 , h表示一個(gè) InvocationHandler 對(duì)象,前面兩個(gè)參數(shù)容易理解,
最后一個(gè)InvocationHandler是什么?
InvocationHandler是一個(gè)接口,官方文檔解釋說,每個(gè)代理的實(shí)例都有一個(gè)與之關(guān)聯(lián)的 InvocationHandler 實(shí)現(xiàn)類,如果代理的方法被調(diào)用,那么代理便會(huì)通知和轉(zhuǎn)發(fā)給內(nèi)部的 InvocationHandler 實(shí)現(xiàn)類,由它決定處理。
public interface InvocationHandler { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable;}
InvocationHandler 內(nèi)部只有一個(gè) invoke() 方法,正是這個(gè)方法決定了怎么樣處理代理傳遞過來的方法調(diào)用。其中參數(shù)proxy表示代理對(duì)象,method表示代理對(duì)象調(diào)用的方法,args表示調(diào)用的方法中的參數(shù)。所以Proxy動(dòng)態(tài)產(chǎn)生的代理會(huì)調(diào)用InvocationHandler實(shí)現(xiàn)類,所以InvocationHandler才是實(shí)際執(zhí)行者。
3、代碼實(shí)例
//抽象主題interface AbstractSubject{ void request();}//真實(shí)主題class RealSubject implements AbstractSubject{ public void request() { System.out.println('訪問真實(shí)主題方法...'); }}//真實(shí)主題class RealSubject1 implements AbstractSubject{ public void request() { System.out.println('訪問真實(shí)主題方法1...'); }}//動(dòng)態(tài)代理類class DynamicProxy implements InvocationHandler { private Object object; public DynamicProxy(Object object) { this.object = object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { before(); Object result = method.invoke(object, args); after(); return result; } private void before() { System.out.println('hello!'); } private void after() { System.out.println('bye!'); }}//測(cè)試類public class TestDynamicProxyPattern { public static void main(String[] args) { AbstractSubject abstractSubject = new RealSubject(); DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject); Proxy.newProxyInstance(abstractSubject.getClass().getClassLoader(), abstractSubject.getClass().getInterfaces() , dynamicProxy); abstractProxy.request(); }}
上述動(dòng)態(tài)代理只需要傳入需要被代理類的對(duì)象(DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject)),然后調(diào)用Proxy類的工廠方法newProxyInstance去動(dòng)態(tài)地創(chuàng)建一個(gè)代理類,最后調(diào)用代理類的方法便實(shí)現(xiàn)了“增強(qiáng)功能”。使用了動(dòng)態(tài)代理之后,無論有多少類多少方法需要增加邏輯,只需要在使用的時(shí)候?qū)㈩悓?duì)象傳入得到代理對(duì)象,然后使用代理對(duì)象調(diào)用需要增強(qiáng)的方法即可。
所以這時(shí)候如果增加一個(gè)實(shí)現(xiàn)抽象主題的真是主題類,比如說叫做RealSubject1,這個(gè)時(shí)候只要只要把該類的對(duì)象傳入動(dòng)態(tài)代理類DynamicProxy中,通過接口又可以實(shí)現(xiàn)接口AbstractSubject的實(shí)現(xiàn)類。
這樣來實(shí)現(xiàn):
AbstractSubject abstractSubject1 = new RealSubject1();DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject1);
4、總結(jié)
1、區(qū)別于靜態(tài)代理的生成代理類,動(dòng)態(tài)代理的代理類通過 Proxy.newInstance() 方法生成。靜態(tài)代理和動(dòng)態(tài)代理的區(qū)別是在于要不要開發(fā)者自己定義 Proxy 類。
2、不管是靜態(tài)代理還是動(dòng)態(tài)代理,代理與被代理者都要實(shí)現(xiàn)接口,還是要是面向接口編程,目的都是增強(qiáng)現(xiàn)有功能。
3、動(dòng)態(tài)代理通過 Proxy 動(dòng)態(tài)生成 proxy class,但是它也指定了一個(gè) InvocationHandler 的實(shí)現(xiàn)類。
4、動(dòng)態(tài)代理也有缺陷,它要求需要代理的對(duì)象必須實(shí)現(xiàn)了某個(gè)接口,而且也不夠靈活,動(dòng)態(tài)代理會(huì)為接口中的聲明的所有方法添加上相同的代理邏輯。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. CSS3實(shí)現(xiàn)動(dòng)態(tài)翻牌效果 仿百度貼吧3D翻牌一次動(dòng)畫特效2. ASP.NET MVC使用正則表達(dá)式驗(yàn)證手機(jī)號(hào)碼3. ASP.NET MVC把數(shù)據(jù)庫中枚舉項(xiàng)的數(shù)字轉(zhuǎn)換成文字4. Python驗(yàn)證的50個(gè)常見正則表達(dá)式5. python re模塊和正則表達(dá)式6. 詳解Python requests模塊7. ajax實(shí)現(xiàn)簡(jiǎn)單登錄頁面8. vue在install時(shí)node-sass@4.14.1 postinstall:node scripts/build.js錯(cuò)誤解決9. 詳解python requests中的post請(qǐng)求的參數(shù)問題10. .NET Core Web APi類庫內(nèi)嵌運(yùn)行的方法
