动态代理:相比于静态代理,动态代理避免了开发人员编写各个繁锁的静态代理类,只需简单地指定一组接口及目标类对象就能动态的获得代理对象。
代理模式:使用代理模式必须要让代理类和目标类实现相同的接口,客户端通过代理类来调用目标方法,代理类会将所有的方法调用分派到目标对象上反射执行,还可以在分派过程中添加”前置通知”和后置处理(如在调用目标方法前校验权限,在调用完目标方法后打印日志等)等功能。
使用动态代理的五大步骤
1. 通过实现InvocationHandler接口来定义自己的InvocationHandler;
2. 通过Proxy.getProxyClass获得动态代理类
3. 通过反射机制获取代理类的构造方法,方法签名为GetConstructor(InvocationHandler.class)
4. 通过构造函数获得代理对象并将定义的InvocationHandler实例对象作为参数传入
5. 通过代理对象调用目标方法
代码如下:
public interface IHello {
void sayHello();
}
public class Hello implements IHello {
public void sayHello() {
// TODO Auto-generated method stub
System.out.println("Hello");
}
}
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class MyInvocationHandler implements InvocationHandler {
private Object target;
public MyInvocationHandler(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// TODO Auto-generated method stub
System.out.println("插入前置通知");
Object rs = method.invoke(target, args);
System.out.println("插入后置通知");
return rs;
}
}
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Proxy;
public class Test {
public static void main(String[] args) throws NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
// TODO Auto-generated method stub
System.getProperties().put(
"sun.misc.ProxyGenerator.saveGeneratedFiles",
true);
Class proxClazz = Proxy.getProxyClass(IHello.class.getClassLoader(),
IHello.class);
Constructor constructor = proxClazz.getConstructor(InvocationHandler.class);
IHello iHello = (IHello) constructor.newInstance(new MyInvocationHandler(new Hello()));
iHello.sayHello();
}
}