Java 動(dòng)態(tài)代理底層原理是基于反射機(jī)制實(shí)現(xiàn)的,其中最重要的是 InvocationHandler 接口,它定義了一個(gè) invoke() 方法,用于實(shí)現(xiàn)對(duì)代理類中各個(gè)方法的調(diào)用,在 invoke() 方法中,可以實(shí)現(xiàn)對(duì)真實(shí)角色的調(diào)用,并進(jìn)行擴(kuò)展,實(shí)現(xiàn)動(dòng)態(tài)代理的效果。

Java動(dòng)態(tài)代理是一種在運(yùn)行時(shí)創(chuàng)建代理類的機(jī)制,動(dòng)態(tài)代理可以在不修改源代碼的情況下,在運(yùn)行時(shí)為某個(gè)接口動(dòng)態(tài)生成實(shí)現(xiàn)類,并且可以攔截接口中的方法調(diào)用,從而實(shí)現(xiàn)一些特殊的功能。
Java 動(dòng)態(tài)代理底層原理是基于反射機(jī)制實(shí)現(xiàn)的,其中最重要的是 InvocationHandler 接口,它定義了一個(gè) invoke() 方法,用于實(shí)現(xiàn)對(duì)代理類中各個(gè)方法的調(diào)用,在 invoke() 方法中,可以實(shí)現(xiàn)對(duì)真實(shí)角色的調(diào)用,并進(jìn)行擴(kuò)展,實(shí)現(xiàn)動(dòng)態(tài)代理的效果。
動(dòng)態(tài)代理的實(shí)現(xiàn)步驟:
1. 創(chuàng)建一個(gè)實(shí)現(xiàn) InvocationHandler 接口的類,它必須實(shí)現(xiàn) invoke() 方法;
2. 創(chuàng)建被代理的真實(shí)對(duì)象;
3. 通過 Proxy 類的 newProxyInstance() 方法創(chuàng)建代理對(duì)象,它需要參數(shù):
(1)ClassLoader:類加載器,它是用于加載代理對(duì)象字節(jié)碼的,和被代理對(duì)象使用相同的類加載器;
(2)Class[]:字節(jié)碼數(shù)組,它是用于讓代理對(duì)象和被代理對(duì)象有相同方法;
(3)InvocationHandler:它是調(diào)用處理器,執(zhí)行代理對(duì)象的方法時(shí),會(huì)觸發(fā)該對(duì)象的 invoke() 方法;
4. 通過代理對(duì)象調(diào)用目標(biāo)方法,實(shí)際上會(huì)轉(zhuǎn)到 invoke() 方法中,在 invoke() 方法中,可以進(jìn)行預(yù)處理、調(diào)用后處理等工作。
反射上一篇文章講過了,這里就是反射的一個(gè)應(yīng)用,這個(gè)最常見的場(chǎng)景就是給代碼加日志,例如在執(zhí)行某個(gè)函數(shù)前將請(qǐng)求體加入到日志中,執(zhí)行后將結(jié)果加入到日志中。這樣可以在不改變?cè)瓉泶a的基礎(chǔ)上來實(shí)現(xiàn)。接下來就使用上一篇反射的例子繼續(xù)擴(kuò)展。
先寫一個(gè)Person的接口:
package ReflectTest;
public interface PersonInterface {
void printName();
void printAge();
}
寫他的實(shí)現(xiàn):
package ReflectTest;
public class Person implements PersonInterface{
private String name = "xiaoming";
private String age = "12";
@Override
public void printName()
{
System.out.println(name);
}
@Override
public void printAge()
{
System.out.println(age);
}
}
然后寫動(dòng)態(tài)代理類:
package ReflectTest;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyHandler implements InvocationHandler {
private Object target;
public ProxyHandler(Object target) {
this.target = target;
}
public Object bind() {
return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("執(zhí)行函數(shù)前加入日志");
Object result = method.invoke(target, args);
System.out.println("執(zhí)行函數(shù)后加入日志");
return result;
}
}
最后測(cè)試一下:
package ReflectTest;
public class main {
public static void main(String[] args) {
ProxyHandler proxyHandler = new ProxyHandler(new Person());
PersonInterface person = (PersonInterface)proxyHandler.bind();
person.printName();
}
}
我們看看結(jié)果在執(zhí)行函數(shù)前后執(zhí)行了其他操作:
SpringAOP其實(shí)原理就類似這種。
