代理分为静态代理和动态代理
静态代理:
通过聚合实现。即代理对象和被代理对象实现相同的接口。代理对象在重写方法时会调用被代理对象的方法。而在调用前后会加入相关逻辑,例如日志,安全检查等。
举例
接口(代理对象和被代理对象都要实现)
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类即可以实现动态代理