代理分爲靜態代理和動態代理
靜態代理:
通過聚合實現。即代理對象和被代理對象實現相同的接口。代理對象在重寫方法時會調用被代理對象的方法。而在調用前後會加入相關邏輯,例如日誌,安全檢查等。
舉例
接口(代理對象和被代理對象都要實現)
package com.fw.proxy;
public class Car implements Moveable{
@Override
public void run() {
System.out.println("car moving...");
}
}
Car對象(被代理對象)
package com.fw.proxy;
public class Car implements Moveable{
@Override
public void run() {
System.out.println("car moving...");
}
}
CarProxy對象(代理對象)
package com.fw.proxy;
public class CarLogProxy implements Moveable{
/**
* 這是靜態代理的實現。動態代理要實現的目的如下:
* 傳入被代理對象的接口之後,由用戶自定義好代理實現的內容,即可完成代理。
* 需要做的工作:
* 動態生成一個代理對象,這個對象對於用戶來說是不可見的,就如同這裏的CarLogProxy一樣。只不過這裏的CarLogProxy
* 是可見的。
*
*
*
*/
private Moveable m;
public CarLogProxy(Moveable m) {
this.m = m;
}
@Override
public void run() {
System.out.println("日誌開始");
m.run();
System.out.println("日誌結束");
}
}
以上是靜態代理
動態代理的原理:
JDK裏動態代理需要根據用戶傳入(接口信息)和自定義(對InvocationHandler的實現)的信息來動態編譯生成一個代理類,拿上面的例子舉例,JDK需要做的就是幫助用戶自動生成CarLogProxy這個類和對象。那麼JDK如何實現呢
第一步,由用戶自定義實現InvocationHandler接口的類
package com.bjsxt.proxy;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class TimeHandler implements InvocationHandler{
//該類的作用就是在調用被代理類的同時在其前後加入需要代理的邏輯,例如日誌,安全檢查等等
private Object target;
public TimeHandler(Object target) {
super();
this.target = target;
}
@Override
public void invoke(Object o, Method m) {
long start = System.currentTimeMillis();
System.out.println("starttime:" + start);
//System.out.println(o.getClass().getName());
try {
m.invoke(target);
} catch (Exception e) {
e.printStackTrace();
}
long end = System.currentTimeMillis();
System.out.println("time:" + (end-start));
}
}
第二步動態編譯生成Proxy對象。該對象負責產生對用戶不可見的代理對象,主要工作就是將字符串編譯成class文件,load到內存之中再創建代理對象
package com.bjsxt.proxy;
import java.io.File;
import java.io.FileWriter;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import javax.tools.JavaCompiler;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import javax.tools.JavaCompiler.CompilationTask;
public class Proxy {
public static Object newProxyInstance(Class infce, InvocationHandler h) throws Exception { //JDK6 Complier API, CGLib, ASM
String methodStr = "";
String rt = "\r\n";
Method[] methods = infce.getMethods();
/*
for(Method m : methods) {
methodStr += "@Override" + rt +
"public void " + m.getName() + "() {" + rt +
" long start = System.currentTimeMillis();" + rt +
" System.out.println(\"starttime:\" + start);" + rt +
" t." + m.getName() + "();" + rt +
" long end = System.currentTimeMillis();" + rt +
" System.out.println(\"time:\" + (end-start));" + rt +
"}";
}
*/
for(Method m : methods) {
methodStr += "@Override" + rt +
"public void " + m.getName() + "() {" + rt +
" try {" + rt +
" Method md = " + infce.getName() + ".class.getMethod(\"" + m.getName() + "\");" + rt +
" h.invoke(this, md);" + rt +
" }catch(Exception e) {e.printStackTrace();}" + rt +
"}";
}
String src =
"package com.bjsxt.proxy;" + rt +
"import java.lang.reflect.Method;" + rt +
"public class $Proxy1 implements " + infce.getName() + "{" + rt +
" public $Proxy1(InvocationHandler h) {" + rt +
" this.h = h;" + rt +
" }" + rt +
" com.bjsxt.proxy.InvocationHandler h;" + rt +
methodStr +
"}";
String fileName =
"d:/src/com/bjsxt/proxy/$Proxy1.java";
File f = new File(fileName);
FileWriter fw = new FileWriter(f);
fw.write(src);
fw.flush();
fw.close();
//compile
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager fileMgr = compiler.getStandardFileManager(null, null, null);
Iterable units = fileMgr.getJavaFileObjects(fileName);
CompilationTask t = compiler.getTask(null, fileMgr, null, null, null, units);
t.call();
fileMgr.close();
//load into memory and create an instance
URL[] urls = new URL[] {new URL("file:/" + "d:/src/")};
URLClassLoader ul = new URLClassLoader(urls);
Class c = ul.loadClass("com.bjsxt.proxy.$Proxy1");
System.out.println(c);
Constructor ctr = c.getConstructor(InvocationHandler.class);
Object m = ctr.newInstance(h);
//m.move();
return m;
}
}
故通過InvocationHandler和Proxy類即可以實現動態代理