JDK动态代理是代理模式一中实现方式,只能代理接口。在不改变原有的业务逻辑统一实现其他业务功能。
JDK动态代理主要由 invocationHandler 接口 和 Proxy 类
1.Proxy类:
static Class getProxyClass (ClassLoaderloader, Class[] interfaces)方法根据传进来的ClassLoader、被代理对象的父接口数组,来动态创建二进制的class文件,在根据创建好的Class 二进制文件,获取创建的动态代理对象。
通过代理类的class对象,获取class对象中参数为invocationHandler的构造方法
判断构造方法中的访问修饰符,如果修饰符不为public,则修改为public将其设为可以访问的
调用构造器中的newInstance方法,参数为invocationHandler,返回代理类的实例
2 invocationHandler 接口:
该接口中只定义一个方法Object invoke(Object proxy, Method method, Object[] args);
Object 代理类实例 Method 被代理的方法 args 为该方法中的参数
动态代理实现步骤
自定义需要被代理的接口和方法
创建一个类来实现invocationHandler接口
通过Proxy的静态方法newProxyInstance(ClassLoaderloader, Class[] interfaces, InvocationHandler h)创建一个代理
public class InvocationHandlerImpl implements InvocationHandler {
/**
* 这个就是我们要代理的真实对象
*/
private Object subject;
/**
* 构造方法,给我们要代理的真实对象赋初值
* @param subject
*/
public InvocationHandlerImpl(Object subject) {
this.subject = subject;
}
/**
* 该方法负责集中处理动态代理类上的所有方法调用。
* 调用处理器根据这三个参数进行预处理或分派到委托类实例上反射执行
* @param proxy 代理类实例
* @param method 被调用的方法对象
* @param args 调用参数
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//在代理真实对象前我们可以添加一些自己的操作
System.out.println("调用之前,------------ ?");
System.out.println("Method:" + method);
//当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用
Object returnValue = method.invoke(subject, args);
//在代理真实对象后我们也可以添加一些自己的操作
System.out.println("调用之后,----------- ?");
return returnValue;
}
}
public static void main(String[] args) {
/*需要被代理的方法*/
Subject realSubject = new RealSubject();
InvocationHandler invocationHandler = new InvocationHandlerImpl(realSubject);
//需要被代理对象的class
ClassLoader classLoader = realSubject.getClass().getClassLoader();
Class[] interfaces = realSubject.getClass().getInterfaces();
Subject subject = (Subject) Proxy.newProxyInstance(classLoader, interfaces, invocationHandler);
System.out.println("代理对象的类型: " + subject.getClass().getName());
/* SayHello为被代理接口中的方法 */
String hi = subject.SayHello("刘炳岗");
System.out.println(hi);
}