上一章讲解了jdk代理类,
1、那么jdk代理模式和CGLIb代理模式有什么区别呢?
jdk缺点:jdk动态代理要求被代理类必须实现接口,而代理类如果没有实现接口则会报错,而报错是因为找不到被代理类的父类!
cglib优点:cglib是通过字节码的方式,可以不需要代理类就能动态代理,cglib动态代理比使用jdk反射代理要更快一点
2、如何实现CGlib代理?
实现MethodInterceptor接口的intercept方法后,所生成的代理方法都调用这个方法。
intercept方法参数:
obj 目标类实例
method 目标方法实例(通过反射获取方法实例)
args 目标方法参数
proxy 代理类实例
该方法的返回值就是目标方法的返回值
3、CGlib引入pom.xml文件
<!-- https://mvnrepository.com/artifact/cglib/cglib -->
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.3.0</version>
</dependency>
java代码实现cglib代理
被代理对象(其实就是个简单的类和方法):
public class RealSubject {
public void send() {
System.out.println("实现主要业务逻辑");
}
}
代理对象:
public class SubjectProxy implements MethodInterceptor{
public Object intercept(Object arg0, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("》》》》》》》实现日志打印,或者其他操作"+method.getName());
Object obj = methodProxy.invokeSuper(arg0, objects);
System.out.println("》》》》》》》实现日志打印,或者其他操作结束"+method.getName());
return obj;
}
}
main方法调用运行代理对象:
public class SubjectMain {
/**
* @param args
*/
public static void main(String[] args) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(RealSubject.class);//继承被代理类
enhancer.setCallback(new SubjectProxy());//设置回调
RealSubject realSubject = (RealSubject) enhancer.create();//生成代理对象
realSubject.send();
}
}
CGLIB会让生成的代理类继承被代理类,并在代理类中对代理方法进行强化处理(前置处理、后置处理等)。在CGLIB底层,其实是借助了ASM这个非常强大的Java字节码生成框架。
我们从上面代码可以看到,代理类对象是由Enhancer类创建的。Enhancer是CGLIB的字节码增强器,可以很方便的对类进行拓展,如上面代码中为类设置的Superclass。
4、到此为止CGLIB动态代理机制就介绍完了,下面给出三种代理方式之间对比
代理方式 | 实现 | 优点 | 缺点 | 特点 |
JDK静态代理 | 代理类与被代理类实现同一接口,并且在代理类中需要硬编码接口 | 实现简单,容易理解 | 代理类需要硬编码接口,在实际应用中可能会导致重复编码,浪费存储空间并且效率很低 | 目前没发现有啥特点 |
JDK动态代理 | 代理类与委托类实现同一接口,主要是通过代理类实现InvocationHandler并重写invoke方法来进行动态代理的,在invoke方法中将对方法进行增强处理 | 不需要硬编码接口,代码复用率高 | 只能够用于被代理类有父类的情况下 | 底层使用反射机制进行方法的调用 |
CGLIB动态代理 | 代理类将被代理类作为自己的父类,并为其中的非final被代理方法创建两个方法,一个是与被代理类的方法签名相同的方法,它在方法中会通过super调用被代理类的方法;另一个是代理类独有的方法。在代理方法中,它会判断是否存在实现了MethodInterceptor接口的对象,若存在则将调用intercept方法对被代理类的方法进行代理 | 可以在运行时对类或者是接口进行增强操作,且被代理类无需实现接口 | 不能对final类以及final方法进行代理 | 底层将方法全部存入一个数组中,通过数组索引直接进行方法调用 |
github项目地址:github项目地址