Java JDK動態(tài)代理實(shí)現(xiàn)原理實(shí)例解析
JDK動態(tài)代理實(shí)現(xiàn)原理
動態(tài)代理機(jī)制
通過實(shí)現(xiàn) InvocationHandler 接口創(chuàng)建自己的調(diào)用處理器
通過為 Proxy 類指定 ClassLoader 對象和一組 interface 來創(chuàng)建動態(tài)代理類
通過反射機(jī)制獲得動態(tài)代理類的構(gòu)造函數(shù),其唯一參數(shù)類型是調(diào)用處理器接口類型
通過構(gòu)造函數(shù)創(chuàng)建動態(tài)代理類實(shí)例,構(gòu)造時(shí)調(diào)用處理器對象作為參數(shù)被傳入
Interface InvocationHandler
該接口中僅定義了一個(gè)方法Object:invoke(Object obj,Method method,Object[] args)。在實(shí)際使用時(shí),第一個(gè)參數(shù)obj一般是指代理類,method是被代理的方法,args為該方法的參數(shù)數(shù)組。這個(gè)抽象方法在代理類中動態(tài)實(shí)現(xiàn)。
Proxy
該類即為動態(tài)代理類
Protected Proxy(InvocationHandler h)
構(gòu)造函數(shù),用于給內(nèi)部的h賦值
Static Class getProxyClass (ClassLoader loader,Class[] interfaces)
獲得一個(gè)代理類,其中l(wèi)oader是類裝載器,interfaces是真實(shí)類所擁有的全部接口的數(shù)組Static Object newProxyInstance(ClassLoader loader,Class[] interfaces,InvocationHandler h)
返回代理類的一個(gè)實(shí)例,返回后的代理類可以當(dāng)作被代理類使用(可使用被代理類的在Subject接口中聲明過的方法)
Dynamic Proxy
它是在運(yùn)行時(shí)生成的class,在生成它時(shí)你必須提供一組interface給它,然后該class就宣稱它實(shí)現(xiàn)了這些 interface。你當(dāng)然可以把該class的實(shí)例當(dāng)作這些interface中的任何一個(gè)來用。當(dāng)然啦,這個(gè)Dynamic Proxy其實(shí)就是一個(gè)Proxy,它不會替你作實(shí)質(zhì)性的工作,在生成它的實(shí)例時(shí)你必須提供一個(gè)handler,由它接管實(shí)際的工作。
代碼示例
創(chuàng)建接口:
/** * @CreateDate: 2019/6/17 14:52 * @Version: 1.0 */public interface BuyService { String buyPhone(); String buyComputer();}
創(chuàng)建實(shí)現(xiàn)類:
public class BuyServiceImpl implements BuyService { @Intercept('buyPhone') @Override public String buyPhone() { try { TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println('==========BuyServiceImpl.class=============' + ' buyPhone'); this.buyComputer(); return 'buy phone'; } @Intercept('buyComputer') @Override public String buyComputer() { try { TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println('==========BuyServiceImpl.class=============' + ' buyComputer'); return 'buy computer'; }}
創(chuàng)建 InvocationHandler:
public class ReflectionHandler implements InvocationHandler { private Object target; public ReflectionHandler(Object target) { this.target = target; } public <T> T getProxy(){ return (T) Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { return method.invoke(target,args); }}
創(chuàng)建啟動類:
public class Bootstrap { public static void main(String[] args) { // 動態(tài)代理實(shí)現(xiàn) ReflectionHandler reflectionHandler = new ReflectionHandler(new BuyServiceImpl()); BuyService proxy = reflectionHandler.getProxy(); String computer = proxy.buyComputer(); String phone = proxy.buyPhone(); System.out.println(computer + 'rn' + phone); }
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. idea導(dǎo)入maven項(xiàng)目的方法2. idea設(shè)置代碼格式化的方法步驟3. 為什么從類內(nèi)部訪問類變量需要“自我”。在Python中?4. ASP 信息提示函數(shù)并作返回或者轉(zhuǎn)向5. Ajax實(shí)現(xiàn)異步加載數(shù)據(jù)6. spring boot 若依系統(tǒng)整合Ueditor部署時(shí)上傳圖片錯誤問題7. ajax請求添加自定義header參數(shù)代碼8. Idea中maven項(xiàng)目實(shí)現(xiàn)登錄驗(yàn)證碼功能9. 教你在 IntelliJ IDEA 中使用 VIM插件的詳細(xì)教程10. 將Oracle 10g內(nèi)置的安全特性用于PHP
