动态代理:相比于静态代理,动态代理避免了开发人员编写各个繁锁的静态代理类,只需简单地指定一组接口及目标类对象就能动态的获得代理对象。
代理模式:使用代理模式必须要让代理类和目标类实现相同的接口,客户端通过代理类来调用目标方法,代理类会将所有的方法调用分派到目标对象上反射执行,还可以在分派过程中添加”前置通知”和后置处理(如在调用目标方法前校验权限,在调用完目标方法后打印日志等)等功能。
使用动态代理的五大步骤
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(); } }