您的位置:首页 > 编程语言 > Java开发

JDK动态代理与CGLIB动态代理

2018-03-01 19:09 246 查看
JDK动态代理

JDK的代理最大的缺点是需要提供接口,在MyBatis的Mapper就是一个接口,它采用的就是JDK的动态代理。

java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。

JDK动态代理中包含一个类和一个接口:

InvocationHandler接口:

public interface InvocationHandler {
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable;
}


参数说明:

Object proxy:指被代理的对象。

Method method:要调用的方法

Object[] args:方法调用时所需要的参数

InvocationHandler接口的子类想象成一个代理的最终操作类

Proxy类:

Proxy类是专门完成代理的操作类,可以通过此类为一个或多个接口动态地生成实现类,此类提供了如下的操作方法:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)  throws IllegalArgumentException


参数说明:

ClassLoader loader:类加载器

Class<>[] interfaces:得到全部的接口

InvocationHandler h:得到InvocationHandler接口的子类实例

动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。

Cglib动态代理

JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。

他需要实现MethodInterceptor接口

public class HelloServiceCglib implements MethodInterceptor {
private Object target;

/**
* 创建代理对象
*
* @param target
* @return
*/
public Object getInstance(Object target) {
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(this.target.getClass());
// 回调方法
enhancer.setCallback(this);
// 创建代理对象
return enhancer.create();
}

@Override
// 回调方法
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable {
System.out.println("我准备说hello了");
proxy.invokeSuper(obj, args);
System.out.println("我说过hello了");
return null;

}

}


这样便能够实现CGLIB的动态代理,在MyBatis中通常在延迟加载的时候才会用到CGLIB的动态代理。

一般说Spring的核心是IOC和AOP,IOC的实现原理是工厂模式加反射技术,而AOP的实现原理就是动态代理,MyBatis中的Mapper接口也是通过JDK动态代理实现。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  mybatis