一、动态代理是通过一个代理对象来调用真实服务对象。动态代理分为两种,一种是JDK反射机制提供的代理,另一种是CGLIB代理。JDK代理需要接口,CGLIB代理不需要接口
二、JDK动态代理
通过Java的反射机制,获取目标对象,反射调用函数。由java.lang.reflect.*提供支持,主要有一下几个步骤:
①:创建接口以及实现类
public interface IReflectService {
public void sayHello();
}
public class ReflectService implements IReflectService {
public void sayHello() {
System.out.println("hello world");
}
}
②:编写一个代理类,提供真实对象的绑定和代理方法,代理类需要实现InvocationHandler接口中的代理方法,当一个对象被绑定后,执行其方法就会进入到代理方法中
public class ReflectProxy implements InvocationHandler {
private Object target;
//绑定对象并返回一个代理类
public Object bind(Object target){
this.target = target;
return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("方法执行前");
Object result = null;
result = method.invoke(target,args);
System.out.println("方法执行后");
return result;
}
}
③:测试
public class Test {
public static void main(String[] args){
ReflectProxy proxy = new ReflectProxy();
IReflectService reflectService = (IReflectService) proxy.bind(new ReflectProxy());
reflectService.sayHello();
}
}
在mybatis中使用Mapper接口就是使用的JDK动态代理方式实现的,并且在方法调用前后可以加入自己想要的东西
三、CGLIB动态代理
为了弥补JDK动态代理必须提供接口的缺陷,我们可以使用开源框架---CGLIB动态代理类
①:前面的ReflectService以及IReflectService都不需要改变
②:实现CGLIB的代理类
public class ReflectProxy implements MethodInterceptor {
private Object target;
public Object bind(Object target){
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass().getSuperclass());
enhancer.setCallback(this);
return enhancer.create();
}
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("方法执行前");
Object result = null;
result = methodProxy.invokeSuper(target,objects);
System.out.println("方法执行后");
return result;
}
}
在Mybatis中延迟加载时使用了CGLIB动态代理