Java JDK動態(tài)代理實現(xiàn)原理實例解析
JDK動態(tài)代理實現(xiàn)原理
動態(tài)代理機制
通過實現(xiàn) InvocationHandler 接口創(chuàng)建自己的調(diào)用處理器
通過為 Proxy 類指定 ClassLoader 對象和一組 interface 來創(chuàng)建動態(tài)代理類
通過反射機制獲得動態(tài)代理類的構(gòu)造函數(shù),其唯一參數(shù)類型是調(diào)用處理器接口類型
通過構(gòu)造函數(shù)創(chuàng)建動態(tài)代理類實例,構(gòu)造時調(diào)用處理器對象作為參數(shù)被傳入
Interface InvocationHandler
該接口中僅定義了一個方法Object:invoke(Object obj,Method method,Object[] args)。在實際使用時,第一個參數(shù)obj一般是指代理類,method是被代理的方法,args為該方法的參數(shù)數(shù)組。這個抽象方法在代理類中動態(tài)實現(xiàn)。
Proxy
該類即為動態(tài)代理類
Protected Proxy(InvocationHandler h)
構(gòu)造函數(shù),用于給內(nèi)部的h賦值
Static Class getProxyClass (ClassLoader loader,Class[] interfaces)
獲得一個代理類,其中l(wèi)oader是類裝載器,interfaces是真實類所擁有的全部接口的數(shù)組
Static Object newProxyInstance(ClassLoader loader,Class[] interfaces,InvocationHandler h)
返回代理類的一個實例,返回后的代理類可以當(dāng)作被代理類使用(可使用被代理類的在Subject接口中聲明過的方法)
Dynamic Proxy
它是在運行時生成的class,在生成它時你必須提供一組interface給它,然后該class就宣稱它實現(xiàn)了這些 interface。你當(dāng)然可以把該class的實例當(dāng)作這些interface中的任何一個來用。當(dāng)然啦,這個Dynamic Proxy其實就是一個Proxy,它不會替你作實質(zhì)性的工作,在生成它的實例時你必須提供一個handler,由它接管實際的工作。
代碼示例
創(chuàng)建接口:
/**
* @CreateDate: 2019/6/17 14:52
* @Version: 1.0
*/
public interface BuyService {
String buyPhone();
String buyComputer();
}
創(chuàng)建實現(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)代理實現(xiàn)
ReflectionHandler reflectionHandler = new ReflectionHandler(new BuyServiceImpl());
BuyService proxy = reflectionHandler.getProxy();
String computer = proxy.buyComputer();
String phone = proxy.buyPhone();
System.out.println(computer + "\r\n" + phone);
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java數(shù)據(jù)結(jié)構(gòu)之HashMap源碼深入分析
Java HashMap是一種基于哈希表實現(xiàn)的鍵值對存儲結(jié)構(gòu),可以實現(xiàn)快速的數(shù)據(jù)查找和存儲。它是線程不安全的,但在單線程環(huán)境中運行效率高,被廣泛應(yīng)用于Java開發(fā)中2023-04-04
Springboot整合多數(shù)據(jù)源代碼示例詳解
這篇文章主要介紹了Springboot整合多數(shù)據(jù)源代碼示例詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-08-08
Springboot整合分頁插件PageHelper步驟解析
這篇文章主要介紹了Springboot整合分頁插件PageHelper步驟解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-06-06
Spring?Cache+Redis緩存數(shù)據(jù)的實現(xiàn)示例
本文主要介紹了Spring?Cache+Redis緩存數(shù)據(jù),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01

