代理在我們?nèi)粘i_(kāi)發(fā)過(guò)程中有著很重要的角色驴一,它可以處理一些日志記錄休雌,權(quán)限控制,事務(wù)等蛔趴。主要分為靜態(tài)代理和動(dòng)態(tài)代理挑辆,他們的主要區(qū)別是有沒(méi)有源文件,靜態(tài)代理是有源文件的孝情,應(yīng)用起來(lái)比較直觀但是稍有浪費(fèi)且不智能鱼蝉,所以我們一般使用的是動(dòng)態(tài)代理,他是在程序運(yùn)行期間動(dòng)態(tài)生成的箫荡,程序加載動(dòng)態(tài)生成的字節(jié)碼生成Class對(duì)象加以運(yùn)用魁亦,靈活性高。
下面我們介紹一下jdk的動(dòng)態(tài)代理羔挡。
首先聲明一個(gè)接口IHello.java
public interface IHello {
void sayHello(String name);
}
在創(chuàng)建一個(gè)接口的具體的實(shí)現(xiàn)類HelloService.java
public class HelloService implements IHello {
@Override
public void sayHello(String name) {
System.out.println("hello "+name);
}
}
使用jdk的動(dòng)態(tài)代理洁奈,代理的邏輯必須寫在InvocationHandler的實(shí)現(xiàn)類中间唉,當(dāng)調(diào)用接口的具體方法時(shí),實(shí)際上是調(diào)用的InvocationHandler中的invoke方法利术,這個(gè)在后面會(huì)有詳細(xì)的解釋呈野。
HelloHandler.java
public class HelloHandler implements InvocationHandler {
/**
* 調(diào)用的具體的對(duì)象
*/
private Object concreteObj;
public HelloHandler(IHello hello) {
concreteObj = hello;
}
/**
* @param proxy 生成的代理對(duì)象的回填
* @param method
* @param args
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("proxy:"+proxy.getClass().getName());
long start = System.currentTimeMillis();
Object o = method.invoke(concreteObj, args);
System.out.println("cost time :" + (System.currentTimeMillis() - start) + "ms");
return o;
}
}
需要的類都準(zhǔn)備好了,我們來(lái)測(cè)試一下
public static void main(String[] args) {
IHello hello = (IHello) Proxy.newProxyInstance(IHello.class.getClassLoader(), new Class[]{IHello.class}, new HelloHandler(new HelloService()));
hello.sayHello("test");
}
結(jié)果為:
proxy:com.sun.proxy.$Proxy0
hello test
cost time :0ms
我們這個(gè)動(dòng)態(tài)代理的作用就是統(tǒng)計(jì)一下方法調(diào)用的時(shí)間印叁。
我們發(fā)現(xiàn)了兩個(gè)問(wèn)題:
- 這個(gè)HelloHandler中invoke方法第一個(gè)參數(shù)proxy是怎么填入的并且他的class的name為何為com.sun.proxy.$Proxy0
- 我調(diào)用的是IHello中的sayHello方法被冒,為啥他會(huì)走到HelloHandler中的invoke方法。
我們帶著這兩個(gè)疑問(wèn)繼續(xù)看起來(lái)轮蜕。
我們先看一下Proxy是如何newProxyInstance的
下面我們看一下生成的字節(jié)碼是怎樣的
public static void main(String[] args) {
createProxyClassFile();
}
public static void createProxyClassFile() {
String name = "ProxyHello";
byte[] data = ProxyGenerator.generateProxyClass(name, new Class[]{IHello.class});
try {
FileOutputStream out = new FileOutputStream(name + ".class");
out.write(data);
out.close();
} catch (Exception e) {
e.printStackTrace();
}
}
創(chuàng)建的文件這這樣的
public final class ProxyHello extends Proxy implements IHello {
private static Method m1;
private static Method m3;
private static Method m2;
private static Method m0;
public ProxyHello(InvocationHandler var1) throws {
super(var1);
}
public final boolean equals(Object var1) throws {
try {
return ((Boolean)super.h.invoke(this, m1, new Object[]{var1})).booleanValue();
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
}
public final void sayHello(String var1) throws {
try {
super.h.invoke(this, m3, new Object[]{var1});
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
}
public final String toString() throws {
try {
return (String)super.h.invoke(this, m2, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
public final int hashCode() throws {
try {
return ((Integer)super.h.invoke(this, m0, (Object[])null)).intValue();
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
static {
try {
m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")});
m3 = Class.forName("com.pajk.homecare.unittest.proxy.IHello").getMethod("sayHello", new Class[]{Class.forName("java.lang.String")});
m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
} catch (NoSuchMethodException var2) {
throw new NoSuchMethodError(var2.getMessage());
} catch (ClassNotFoundException var3) {
throw new NoClassDefFoundError(var3.getMessage());
}
}
}
根據(jù)這個(gè)文件昨悼,我們看一下,當(dāng)調(diào)用代理的sayHello方法時(shí)跃洛,調(diào)用的是handler的invoke方法率触,傳入的第一個(gè)參數(shù)是this,所以就是Proxy的實(shí)例了汇竭。所以上面提出的兩個(gè)問(wèn)題得解葱蝗。