CGLIB 动态代理及其原理分析

一、简介

  CGLIB,即 Code Generation Library,是一个强大的、高性能的代码生成库。它可以在运行期扩展 Java 类与实现 Java 接口(JDK 动态代理只能用于接口),它被许多 AOP 框架广泛地使用,如为 Spring AOP 提供方法的 interception(拦截),被 Hibernate 用来代理单端(多对一和一对一)关联(延迟提取集合使用的另一种机制)。本文使用的 CGLIB 依赖如下:

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.3.0</version>
</dependency>

二、CGLIB 原理

2.1 基本原理

  CGLIB 通过动态生成一个需要被代理类的子类(即被代理类作为父类),该子类重写被代理类的所有不是 final 修饰的方法,并在子类中采用方法拦截的技术拦截父类所有的方法调用,进而织入横切逻辑。此外,因为 CGLIB 采用整型变量建立了方法索引,这比使用 JDK 动态代理更快(使用 Java 反射技术创建代理类的实例)。

  在底层实现上,CGLIB 使用字节码处理框架 ASM,该框架用于转换字节码并生成新的类。但是不鼓励直接使用 ASM,因为它要求对于 JVM 的内部结构包括 class 文件的格式和 JVM 指令都很熟悉,若一旦出现错误,会导致 JVM 崩溃级别的异常。

  虽然说 CGLIB 与 JDK 动态代理相比,具有更好的优势,当并不是说使用 CGLIB 会更好,这里先总结下 CGLIB 和 JDK 动态代理之间的区别:

  • JDK 动态代理只能对接口进行代理,不能对普通的类进行代理,这是因为 JDK 动态代理生成的代理类,其父类是 Proxy,且 Java 不支持类的多继承。
  • CGLIB 能够代理接口和普通的类,但是被代理的类不能被 final 修饰,且接口中的方法不能使用 final 修饰。
  • JDK 动态代理使用 Java 反射技术进行操作,在生成类上更高效。
  • CGLIB 使用 ASM 框架直接对字节码进行修改,使用了 FastClass 的特性。在某些情况下,类的方法执行会比较高效。
2.2 包结构
  • net.sf.cglib.core:底层字节码处理类,它们大部分与ASM有关系,在使用者角度来看不需要过多关注此包。
  • net.sf.cglib.transform:编译期或运行期类和类文件的转换。
  • net.sf.cglib.proxy:实现创建代理和方法拦截器的类。
  • net.sf.cglib.reflect:反射相关工具类。
  • net.sf.cglib.util:集合排序等工具类。
  • net.sf.cglib.beans:Java Bean 相关的工具类。
2.3 CGLIB 原理分析

  在分析 CGLIB 动态代理的原理前,先创建一个代理类的父类,如下所示:

public class CglibSampleClass {
    public String hello(String name) {
        return String.format("%s say hi to you!", name);
    }
}

  接下来开启 CGLIB 的 debug 模式,并使其生成的类输出到指定的目录。

public class DebuggingCglibDemo {

    private static final String METHOD_NAME = "hello";

    public static void main(String[] args) throws Exception {
        String location = DebuggingCglibDemo.class.getResource("").getPath().replaceAll("%20"," ") + "debugging/";
        System.out.println("location -> " + location);
        // 设置 CGLIB 的 debug 输出位置
        System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, location);

        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(CglibSampleClass.class);
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object obj, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                Object result;
                if (METHOD_NAME.equals(method.getName())) {
                    System.out.println("Before invoking hello...");
                    result = methodProxy.invokeSuper(obj, objects);
                    System.out.println("After invoking hello...");
                } else {
                    result = methodProxy.invokeSuper(obj, objects);
                }
                return result;
            }
        });
        CglibSampleClass sampleClass = (CglibSampleClass) enhancer.create();
        System.out.println(sampleClass.hello("Judy"));
    }
}

  输出结果如下所示:

location -> /D:/IntelliJ IDEA 2019.3.3/idea_workplace/blog-data/java-cglib/target/classes/club/wadreamer/cglib/debugging/
CGLIB debugging enabled, writing to '/D:/IntelliJ IDEA 2019.3.3/idea_workplace/blog-data/java-cglib/target/classes/club/wadreamer/cglib/debugging/'
Before invoking hello...
After invoking hello...
Judy say hi to you!

  生成的目录结构如下所示:

在这里插入图片描述
  接下来,先看下生成的代理类的直接子类CglibSampleClass$$EnhancerByCGLIB$$f9a62642,其继承了 CglibSampleClass 类(被代理类),并实现了 net.sf.cglib.proxy.Factory 接口。

public class CglibSampleClass$$EnhancerByCGLIB$$f9a62642 extends CglibSampleClass implements Factory {
    private boolean CGLIB$BOUND;
    public static Object CGLIB$FACTORY_DATA;
    private static final ThreadLocal CGLIB$THREAD_CALLBACKS;
    private static final Callback[] CGLIB$STATIC_CALLBACKS;
    private MethodInterceptor CGLIB$CALLBACK_0;
    private static Object CGLIB$CALLBACK_FILTER;

    /* 
     * 以静态变量保存了指向代理方法的引用
     */
    private static final Method CGLIB$hello$0$Method; // 直接指向被代理类的方法
    private static final MethodProxy CGLIB$hello$0$Proxy; // CGLIB 生成的方法代理

    private static final Object[] CGLIB$emptyArgs;

    private static final Method CGLIB$equals$1$Method; // 直接指向被代理类的方法
    private static final MethodProxy CGLIB$equals$1$Proxy; // CGLIB 生成的方法代理

    private static final Method CGLIB$toString$2$Method; // 直接指向被代理类的方法
    private static final MethodProxy CGLIB$toString$2$Proxy; // CGLIB 生成的方法代理

    private static final Method CGLIB$hashCode$3$Method; // 直接指向被代理类的方法
    private static final MethodProxy CGLIB$hashCode$3$Proxy; // CGLIB 生成的方法代理

    private static final Method CGLIB$clone$4$Method; // 直接指向被代理类的方法
    private static final MethodProxy CGLIB$clone$4$Proxy; // CGLIB 生成的方法代理

    // ================================== 通过静态代码块初始化上面用到的静态变量,主要是使用到反射 ==================================
    static {
        CGLIB$STATICHOOK1();
    }
    static void CGLIB$STATICHOOK1() {
        CGLIB$THREAD_CALLBACKS = new ThreadLocal();
        CGLIB$emptyArgs = new Object[0];
        Class var0 = Class.forName("club.wadreamer.cglib.CglibSampleClass$$EnhancerByCGLIB$$f9a62642");
        Class var1;

        Method[] var10000 = ReflectUtils.findMethods(new String[]{
            "equals", 
            "(Ljava/lang/Object;)Z", 
            "toString", 
            "()Ljava/lang/String;", 
            "hashCode", 
            "()I", "clone", 
            "()Ljava/lang/Object;"}, 
            (var1 = Class.forName("java.lang.Object")).getDeclaredMethods());
        CGLIB$equals$1$Method = var10000[0];
        CGLIB$equals$1$Proxy = MethodProxy.create(var1, var0, "(Ljava/lang/Object;)Z", "equals", "CGLIB$equals$1");

        CGLIB$toString$2$Method = var10000[1];
        CGLIB$toString$2$Proxy = MethodProxy.create(var1, var0, "()Ljava/lang/String;", "toString", "CGLIB$toString$2");

        CGLIB$hashCode$3$Method = var10000[2];
        CGLIB$hashCode$3$Proxy = MethodProxy.create(var1, var0, "()I", "hashCode", "CGLIB$hashCode$3");

        CGLIB$clone$4$Method = var10000[3];
        CGLIB$clone$4$Proxy = MethodProxy.create(var1, var0, "()Ljava/lang/Object;", "clone", "CGLIB$clone$4");

        CGLIB$hello$0$Method = ReflectUtils.findMethods(new String[]{
            "hello", 
            "(Ljava/lang/String;)Ljava/lang/String;"}, 
            (var1 = Class.forName("club.wadreamer.cglib.CglibSampleClass")).getDeclaredMethods())[0];
        CGLIB$hello$0$Proxy = MethodProxy.create(var1, var0, "(Ljava/lang/String;)Ljava/lang/String;", "hello", "CGLIB$hello$0");
    }

    // ========================================================================================================================

    // ================================== 代理方法、equals、toString、hashCode、clone 方法的调用 ===================================

    // 直接通过调用被代理类的方法
    final String CGLIB$hello$0(String var1) {
        return super.hello(var1);
    }

    // 使用 Callback 实例进行方法代理的回调
    public final String hello(String var1) {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (var10000 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }

        // 若找不到 Callback 实例,则直接调用被代理的方法
        return var10000 != null ? (String)var10000.intercept(this, CGLIB$hello$0$Method, new Object[]{var1}, CGLIB$hello$0$Proxy) : super.hello(var1);
    }

    final boolean CGLIB$equals$1(Object var1) {
        return super.equals(var1);
    }

    public final boolean equals(Object var1) {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (var10000 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }

        if (var10000 != null) {
            Object var2 = var10000.intercept(this, CGLIB$equals$1$Method, new Object[]{var1}, CGLIB$equals$1$Proxy);
            return var2 == null ? false : (Boolean)var2;
        } else {
            return super.equals(var1);
        }
    }

    final String CGLIB$toString$2() {
        return super.toString();
    }

    public final String toString() {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (var10000 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }

        return var10000 != null ? (String)var10000.intercept(this, CGLIB$toString$2$Method, CGLIB$emptyArgs, CGLIB$toString$2$Proxy) : super.toString();
    }

    final int CGLIB$hashCode$3() {
        return super.hashCode();
    }

    public final int hashCode() {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (var10000 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }

        if (var10000 != null) {
            Object var1 = var10000.intercept(this, CGLIB$hashCode$3$Method, CGLIB$emptyArgs, CGLIB$hashCode$3$Proxy);
            return var1 == null ? 0 : ((Number)var1).intValue();
        } else {
            return super.hashCode();
        }
    }

    final Object CGLIB$clone$4() throws CloneNotSupportedException {
        return super.clone();
    }

    protected final Object clone() throws CloneNotSupportedException {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (var10000 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }

        return var10000 != null ? var10000.intercept(this, CGLIB$clone$4$Method, CGLIB$emptyArgs, CGLIB$clone$4$Proxy) : super.clone();
    }

    // ========================================================================================================================


    // 基于整形数字索引,快速查找对应的方法代理
    public static MethodProxy CGLIB$findMethodProxy(Signature var0) {
        String var10000 = var0.toString();
        switch(var10000.hashCode()) {
        case -508378822:
            if (var10000.equals("clone()Ljava/lang/Object;")) {
                return CGLIB$clone$4$Proxy;
            }
            break;
        case 848333779:
            if (var10000.equals("hello(Ljava/lang/String;)Ljava/lang/String;")) {
                return CGLIB$hello$0$Proxy;
            }
            break;
        case 1826985398:
            if (var10000.equals("equals(Ljava/lang/Object;)Z")) {
                return CGLIB$equals$1$Proxy;
            }
            break;
        case 1913648695:
            if (var10000.equals("toString()Ljava/lang/String;")) {
                return CGLIB$toString$2$Proxy;
            }
            break;
        case 1984935277:
            if (var10000.equals("hashCode()I")) {
                return CGLIB$hashCode$3$Proxy;
            }
        }

        return null;
    }

    public CglibSampleClass$$EnhancerByCGLIB$$f9a62642() {
        CGLIB$BIND_CALLBACKS(this);
    }

    public static void CGLIB$SET_THREAD_CALLBACKS(Callback[] var0) {
        CGLIB$THREAD_CALLBACKS.set(var0);
    }

    public static void CGLIB$SET_STATIC_CALLBACKS(Callback[] var0) {
        CGLIB$STATIC_CALLBACKS = var0;
    }
    
    private static final void CGLIB$BIND_CALLBACKS(Object var0) {
        CglibSampleClass$$EnhancerByCGLIB$$f9a62642 var1 = (CglibSampleClass$$EnhancerByCGLIB$$f9a62642)var0;
        if (!var1.CGLIB$BOUND) {
            var1.CGLIB$BOUND = true;
            Object var10000 = CGLIB$THREAD_CALLBACKS.get();
            if (var10000 == null) {
                var10000 = CGLIB$STATIC_CALLBACKS;
                if (var10000 == null) {
                    return;
                }
            }

            var1.CGLIB$CALLBACK_0 = (MethodInterceptor)((Callback[])var10000)[0];
        }

    }

    // ============================= 接下来实现的都是 Factory 接口中的方法 =========================
    public Object newInstance(Callback[] var1) {
        CGLIB$SET_THREAD_CALLBACKS(var1);
        CglibSampleClass$$EnhancerByCGLIB$$f9a62642 var10000 = new CglibSampleClass$$EnhancerByCGLIB$$f9a62642();
        CGLIB$SET_THREAD_CALLBACKS((Callback[])null);
        return var10000;
    }

    public Object newInstance(Callback var1) {
        CGLIB$SET_THREAD_CALLBACKS(new Callback[]{var1});
        CglibSampleClass$$EnhancerByCGLIB$$f9a62642 var10000 = new CglibSampleClass$$EnhancerByCGLIB$$f9a62642();
        CGLIB$SET_THREAD_CALLBACKS((Callback[])null);
        return var10000;
    }

    public Object newInstance(Class[] var1, Object[] var2, Callback[] var3) {
        CGLIB$SET_THREAD_CALLBACKS(var3);
        CglibSampleClass$$EnhancerByCGLIB$$f9a62642 var10000 = new CglibSampleClass$$EnhancerByCGLIB$$f9a62642;
        switch(var1.length) {
        case 0:
            var10000.<init>();
            CGLIB$SET_THREAD_CALLBACKS((Callback[])null);
            return var10000;
        default:
            throw new IllegalArgumentException("Constructor not found");
        }
    }

    public Callback getCallback(int var1) {
        CGLIB$BIND_CALLBACKS(this);
        MethodInterceptor var10000;
        switch(var1) {
        case 0:
            var10000 = this.CGLIB$CALLBACK_0;
            break;
        default:
            var10000 = null;
        }

        return var10000;
    }

    public void setCallback(int var1, Callback var2) {
        switch(var1) {
        case 0:
            this.CGLIB$CALLBACK_0 = (MethodInterceptor)var2;
        default:
        }
    }

    public Callback[] getCallbacks() {
        CGLIB$BIND_CALLBACKS(this);
        return new Callback[]{this.CGLIB$CALLBACK_0};
    }

    public void setCallbacks(Callback[] var1) {
        this.CGLIB$CALLBACK_0 = (MethodInterceptor)var1[0];
    }
}

  从上述代码可以看到,被代理类的子类中有较多的变量,这些变量通过静态代码块基于反射进行初始化。此外,与 JDK 动态代理相比,CGLIB 代理类的代码更多,因此,类的生成效率较低。

  较重要的代码已在类中注释,其他的方法可通过方法名得知其作用,这里只需要注意两点内容即可。

  1) 下面两个静态变量均指向 hello() 方法,CGLIB$hello$0$Method 直接指向被代理类的方法,CGLIB$hello$0$Proxy 指向 CGLIB 生成的方法代理。其他指向 toString()hashCode()equals()clone() 方法的静态变量跟下面的两个变量类似。

private static final Method CGLIB$hello$0$Method;
private static final MethodProxy CGLIB$hello$0$Proxy;

  2) toString()hashCode()equals()clone() 通过直接指向被代理类方法的方式和使用方法代理回调的方式与下面两个方法的实现过程类似。

	// 直接通过调用被代理类的方法
    final String CGLIB$hello$0(String var1) {
        return super.hello(var1);
    }

    // 使用 Callback 实例进行方法代理的回调
    public final String hello(String var1) {
        MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
        if (var10000 == null) {
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_0;
        }

        // 若找不到 Callback 实例,则直接调用被代理的方法
        return var10000 != null ? (String)var10000.intercept(this, CGLIB$hello$0$Method, new Object[]{var1}, CGLIB$hello$0$Proxy) : super.hello(var1);
    }

  接下来再看下 MethodProxyMethodProxy#invoke()MethodProxy#invokeSuper() 方法,这两个方法分别调用了 FastClassInfo的实例 fci.f1fci.f2

在这里插入图片描述
  此外,SampleClass$$EnhancerByCGLIB$$7af3324f$$FastClassByCGLIB$$9d50f44d 对应于被代理类的子类,SampleClass$$FastClassByCGLIB$$96cc97c1 对应于被代理类。接下来展开SampleClass$$EnhancerByCGLIB$$7af3324f$$FastClassByCGLIB$$9d50f44d 类。

public class CglibSampleClass$$EnhancerByCGLIB$$f9a62642$$FastClassByCGLIB$$6efeff76 extends FastClass {
    public CglibSampleClass$$EnhancerByCGLIB$$f9a62642$$FastClassByCGLIB$$6efeff76(Class var1) {
        super(var1);
    }

    public int getIndex(Signature var1) {
        String var10000 = var1.toString();
        switch(var10000.hashCode()) {
        case -2055565910:
            if (var10000.equals("CGLIB$SET_THREAD_CALLBACKS([Lnet/sf/cglib/proxy/Callback;)V")) {
                return 12;
            }
            break;
        case -1882565338:
            if (var10000.equals("CGLIB$equals$1(Ljava/lang/Object;)Z")) {
                return 17;
            }
            break;
        case -1457535688:
            if (var10000.equals("CGLIB$STATICHOOK1()V")) {
                return 15;
            }
            break;
        case -1411842725:
            if (var10000.equals("CGLIB$hashCode$3()I")) {
                return 18;
            }
            break;
        case -894172689:
            if (var10000.equals("newInstance(Lnet/sf/cglib/proxy/Callback;)Ljava/lang/Object;")) {
                return 6;
            }
            break;
        case -879968516:
            if (var10000.equals("CGLIB$hello$0(Ljava/lang/String;)Ljava/lang/String;")) {
                return 19;
            }
            break;
        case -623122092:
            if (var10000.equals("CGLIB$findMethodProxy(Lnet/sf/cglib/core/Signature;)Lnet/sf/cglib/proxy/MethodProxy;")) {
                return 14;
            }
            break;
        case -508378822:
            if (var10000.equals("clone()Ljava/lang/Object;")) {
                return 3;
            }
            break;
        case -419626537:
            if (var10000.equals("setCallbacks([Lnet/sf/cglib/proxy/Callback;)V")) {
                return 9;
            }
            break;
        case 560567118:
            if (var10000.equals("setCallback(ILnet/sf/cglib/proxy/Callback;)V")) {
                return 8;
            }
            break;
        case 811063227:
            if (var10000.equals("newInstance([Ljava/lang/Class;[Ljava/lang/Object;[Lnet/sf/cglib/proxy/Callback;)Ljava/lang/Object;")) {
                return 5;
            }
            break;
        case 848333779:
            if (var10000.equals("hello(Ljava/lang/String;)Ljava/lang/String;")) {
                return 7;
            }
            break;
        case 973717575:
            if (var10000.equals("getCallbacks()[Lnet/sf/cglib/proxy/Callback;")) {
                return 11;
            }
            break;
        case 1221173700:
            if (var10000.equals("newInstance([Lnet/sf/cglib/proxy/Callback;)Ljava/lang/Object;")) {
                return 4;
            }
            break;
        case 1230699260:
            if (var10000.equals("getCallback(I)Lnet/sf/cglib/proxy/Callback;")) {
                return 10;
            }
            break;
        case 1306468936:
            if (var10000.equals("CGLIB$toString$2()Ljava/lang/String;")) {
                return 16;
            }
            break;
        case 1584330438:
            if (var10000.equals("CGLIB$SET_STATIC_CALLBACKS([Lnet/sf/cglib/proxy/Callback;)V")) {
                return 13;
            }
            break;
        case 1800494055:
            if (var10000.equals("CGLIB$clone$4()Ljava/lang/Object;")) {
                return 20;
            }
            break;
        case 1826985398:
            if (var10000.equals("equals(Ljava/lang/Object;)Z")) {
                return 0;
            }
            break;
        case 1913648695:
            if (var10000.equals("toString()Ljava/lang/String;")) {
                return 1;
            }
            break;
        case 1984935277:
            if (var10000.equals("hashCode()I")) {
                return 2;
            }
        }

        return -1;
    }

    public int getIndex(String var1, Class[] var2) {
        switch(var1.hashCode()) {
        case -1776922004:
            if (var1.equals("toString")) {
                switch(var2.length) {
                case 0:
                    return 1;
                }
            }
            break;
        case -1295482945:
            if (var1.equals("equals")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("java.lang.Object")) {
                        return 0;
                    }
                }
            }
            break;
        case -1053468136:
            if (var1.equals("getCallbacks")) {
                switch(var2.length) {
                case 0:
                    return 11;
                }
            }
            break;
        case -124978609:
            if (var1.equals("CGLIB$equals$1")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("java.lang.Object")) {
                        return 17;
                    }
                }
            }
            break;
        case -60403779:
            if (var1.equals("CGLIB$SET_STATIC_CALLBACKS")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("[Lnet.sf.cglib.proxy.Callback;")) {
                        return 13;
                    }
                }
            }
            break;
        case -29025555:
            if (var1.equals("CGLIB$hashCode$3")) {
                switch(var2.length) {
                case 0:
                    return 18;
                }
            }
            break;
        case 85179481:
            if (var1.equals("CGLIB$SET_THREAD_CALLBACKS")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("[Lnet.sf.cglib.proxy.Callback;")) {
                        return 12;
                    }
                }
            }
            break;
        case 94756189:
            if (var1.equals("clone")) {
                switch(var2.length) {
                case 0:
                    return 3;
                }
            }
            break;
        case 99162322:
            if (var1.equals("hello")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("java.lang.String")) {
                        return 7;
                    }
                }
            }
            break;
        case 147696667:
            if (var1.equals("hashCode")) {
                switch(var2.length) {
                case 0:
                    return 2;
                }
            }
            break;
        case 161998109:
            if (var1.equals("CGLIB$STATICHOOK1")) {
                switch(var2.length) {
                case 0:
                    return 15;
                }
            }
            break;
        case 495524492:
            if (var1.equals("setCallbacks")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("[Lnet.sf.cglib.proxy.Callback;")) {
                        return 9;
                    }
                }
            }
            break;
        case 1154623345:
            if (var1.equals("CGLIB$findMethodProxy")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("net.sf.cglib.core.Signature")) {
                        return 14;
                    }
                }
            }
            break;
        case 1543336189:
            if (var1.equals("CGLIB$toString$2")) {
                switch(var2.length) {
                case 0:
                    return 16;
                }
            }
            break;
        case 1811874389:
            if (var1.equals("newInstance")) {
                switch(var2.length) {
                case 1:
                    String var10001 = var2[0].getName();
                    switch(var10001.hashCode()) {
                    case -845341380:
                        if (var10001.equals("net.sf.cglib.proxy.Callback")) {
                            return 6;
                        }
                        break;
                    case 1730110032:
                        if (var10001.equals("[Lnet.sf.cglib.proxy.Callback;")) {
                            return 4;
                        }
                    }
                case 2:
                default:
                    break;
                case 3:
                    if (var2[0].getName().equals("[Ljava.lang.Class;") && var2[1].getName().equals("[Ljava.lang.Object;") && var2[2].getName().equals("[Lnet.sf.cglib.proxy.Callback;")) {
                        return 5;
                    }
                }
            }
            break;
        case 1817099975:
            if (var1.equals("setCallback")) {
                switch(var2.length) {
                case 2:
                    if (var2[0].getName().equals("int") && var2[1].getName().equals("net.sf.cglib.proxy.Callback")) {
                        return 8;
                    }
                }
            }
            break;
        case 1891304123:
            if (var1.equals("CGLIB$hello$0")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("java.lang.String")) {
                        return 19;
                    }
                }
            }
            break;
        case 1905679803:
            if (var1.equals("getCallback")) {
                switch(var2.length) {
                case 1:
                    if (var2[0].getName().equals("int")) {
                        return 10;
                    }
                }
            }
            break;
        case 1951977610:
            if (var1.equals("CGLIB$clone$4")) {
                switch(var2.length) {
                case 0:
                    return 20;
                }
            }
        }

        return -1;
    }

    public int getIndex(Class[] var1) {
        switch(var1.length) {
        case 0:
            return 0;
        default:
            return -1;
        }
    }

    public Object invoke(int var1, Object var2, Object[] var3) throws InvocationTargetException {
        f9a62642 var10000 = (f9a62642)var2;
        int var10001 = var1;

        try {
            switch(var10001) {
            case 0:
                return new Boolean(var10000.equals(var3[0]));
            case 1:
                return var10000.toString();
            case 2:
                return new Integer(var10000.hashCode());
            case 3:
                return var10000.clone();
            case 4:
                return var10000.newInstance((Callback[])var3[0]);
            case 5:
                return var10000.newInstance((Class[])var3[0], (Object[])var3[1], (Callback[])var3[2]);
            case 6:
                return var10000.newInstance((Callback)var3[0]);
            case 7:
                return var10000.hello((String)var3[0]);
            case 8:
                var10000.setCallback(((Number)var3[0]).intValue(), (Callback)var3[1]);
                return null;
            case 9:
                var10000.setCallbacks((Callback[])var3[0]);
                return null;
            case 10:
                return var10000.getCallback(((Number)var3[0]).intValue());
            case 11:
                return var10000.getCallbacks();
            case 12:
                f9a62642.CGLIB$SET_THREAD_CALLBACKS((Callback[])var3[0]);
                return null;
            case 13:
                f9a62642.CGLIB$SET_STATIC_CALLBACKS((Callback[])var3[0]);
                return null;
            case 14:
                return f9a62642.CGLIB$findMethodProxy((Signature)var3[0]);
            case 15:
                f9a62642.CGLIB$STATICHOOK1();
                return null;
            case 16:
                return var10000.CGLIB$toString$2();
            case 17:
                return new Boolean(var10000.CGLIB$equals$1(var3[0]));
            case 18:
                return new Integer(var10000.CGLIB$hashCode$3());
            case 19:
                return var10000.CGLIB$hello$0((String)var3[0]);
            case 20:
                return var10000.CGLIB$clone$4();
            }
        } catch (Throwable var4) {
            throw new InvocationTargetException(var4);
        }

        throw new IllegalArgumentException("Cannot find matching method/constructor");
    }

    public Object newInstance(int var1, Object[] var2) throws InvocationTargetException {
        f9a62642 var10000 = new f9a62642;
        f9a62642 var10001 = var10000;
        int var10002 = var1;

        try {
            switch(var10002) {
            case 0:
                var10001.<init>();
                return var10000;
            }
        } catch (Throwable var3) {
            throw new InvocationTargetException(var3);
        }

        throw new IllegalArgumentException("Cannot find matching method/constructor");
    }

    public int getMaxIndex() {
        return 20;
    }
}

  可以看到,该类中大部分都是 switch-case 块,该类通过为方法的调用添加基于整数数字的索引,来减少反射调用的时间,从而将反射调用转化为直接调用。总而言之,invokeSuper() 方法的使用流程如下:

  • 通过 MethodProxy#init() 方法,用当前方法的 Signature 签名构建两个 FastClass 实例和当前方法对应的 index,并存放在 FastClassInfo 中。
  • 通过 FastClassInfo 中的 FastClass 实例和 index,在 FastClass 中找到对应的方法(在基于整数索引 indexswitch-case 中查找)进行直接调用。

三、总结

  CGLIB 在类的生成阶段所做的操作会相对耗时,且生成的类的数目较多,会占据大量的元空间的内存(永久代)。但是,被代理类的子类一旦生成,之后的方法调用则变成搜索方法索引和直接调用,则在 switch-case 块不大,且当前调用方法的 indexswitch-case 块的前部时,能够获得比 JDK 反射更高的效率。

  • 12
    点赞
  • 51
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值