JDK 与 Cglib 的使用和对比

Spring AOP 依靠 JDK 和 CGLib 进行动态代理实现。在此对两种实现方式的一些知识进行整理。

JDK

使用示例

/**
 * 需要被代理的接口
 */
interface Iinterface {

    String proxyMethod(String gift);
}


/**
 * 实现 InvocationHandler 接口,对 invoke 方法进行重写
 */
class MyHandler implements  {

    /**
     * @param proxy 生成的代理实例
     * @param method 被代理类的方法
     * @param args  传入的参数列表
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        // 对应被代理的方法
        if (method.getName().equals("proxyMethod")) {
            System.out.println("接受到参数:" + args[0]);
            return "返回值";
        }
        return null;
    }
}

public class Main {
    public static void main(String[] args) {
        Iinterface proxy = (Iinterface) Proxy.newProxyInstance(
                Thread.currentThread().getContextClassLoader(),
                new Class[]{Iinterface.class},
                new MyHandler()
        );
        String res = proxy.proxyMethod("传入的参数");
        System.out.println(res);
    }
}

image

可见代理成功。

概括一下,动态代理的方式一般为:

  • 继承 InvocationHandler ,重写方法 invoke

  • 执行 Proxy.newProxyInstance 生成动态代理类

invoke 方法

由上可以看出,proxy 成功对 Iinterface 接口进行代理,但是在使用时,我们并未见到 InvocationHandler 中 invoke 方法的调用,动态代理是如何执行 invoke 的呢?

采用其他资料生成的案例,以下是代理类的反编译代码。

参考链接:链接1 链接2

首先来看看代理类的继承关系:

image

可以看到代理类继承了 Proxy ,再来看看代理类中的方法调用,其中 teach() 是被代理接口的方法声明,内部知识简单地调用了父类即 Proxy 的 h 属性的 invoke 方法

image

再回看 Proxy 类的属性

image

可以猜测,对被代理方法进行调用时,会转而由被代理类中继承自 Proxy 的 InvocationHandler 执行,从而 invoke方法被调用。再来看看调用的 newProxyInstance 的逻辑,代码进行了省略,添加了简单注释

public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h){
        // 创建代理类$Proxy0,实现了传入的 intfs 接口,并继承了 Proxy
        Class<?> cl = getProxyClass0(loader, intfs);
        //...
        // 获取构造方法
        final Constructor<?> cons = cl.getConstructor(constructorParams);
        //...
        //调用具有 InvocationHandler 的构造方法,创建代理类并返回
        return cons.newInstance(new Object[]{h});
        // ...
    }

目前还有一个问题没搞明白:调用 invoke 时,参数中的 method 能够争取匹配到调用的方法,该匹配是如何实现的?

image

CGLib

使用示例

class Hello {

    public String hello(String arg) {
        System.out.println("获取到参数: " + arg);
        return "返回参数";
    }

    /**
     * 通过 final 修饰,该方法不能被子类覆盖,因此 CGLib 无法代理
     */
    final public String helloFinal() {
        System.out.println("helloFinal");
        return null;
    }
}

class MyInterceptor implements MethodInterceptor {
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("代理执行前");
        String res = (String) methodProxy.invokeSuper(o, objects);
        System.out.println(res);
        System.out.println("代理执行后");
        return null;
    }
}

public class Main {
    public static void main(String[] args) {
        // 通过CGLIB动态代理获取代理对象的过程
        Enhancer enhancer = new Enhancer();
        // 设置enhancer对象的父类
        enhancer.setSuperclass(Hello.class);
        // 设置enhancer的回调对象
        enhancer.setCallback(new MyInterceptor());
        // 创建代理对象
        Hello proxy= (Hello)enhancer.create();
        // 通过代理对象调用目标方法
        proxy.hello("参数");
    }
}

image

概括一下,CGLib 动态代理的方式一般为:

  • 构造 Enhancer
  • 设置代理对象(作为父类)
  • 设置代理策略
  • 创建代理对象

Callback

参考链接:https://blog.csdn.net/zhang6622056/article/details/87286498

Callback 可以理解成生成的代理类的方法被调用时会执行的逻辑,具有以下六种方式:

  • NoOp:不做任何操作

  • FixedValue:要求实现接口的 loadObjecd 方法,重写了被代理类的响应方法,同时,要求返回值和方法返回值相同,否则会抛出类型转换异常。此方式看不到人喝原方法的信息,也无法调用原方法。

  • MethodInterceptor:类似 AOP 的环绕增强,代理类的方法调用都会转入执行该接口的 intercept 方法。需要执行原方法可以使用参数 method 进行反射调用,或者使用参数 proxy(proxy会快一些)

  • InvocationHandler:类似 MethodInterceptor,若自定义该接口的 invoke 方法,需要注意参数 methodinvoke方法,会无限循环调用

  • LazyLoader:调用时,返回一个代理对象并存储负责所有的该代理类调用,类似 Spring 的 singleton

  • Dispatcher:每次调用都会返回一个新的代理类,类似 Spring 的 prototye

JDK 与 Cglib 的对比

JDK Cglib
代理目标 接口 类(被代理类会作为父类,无法处理 final)
实现 反射机制 运行过程中修改被代理类的 class 文件字节码
Spring采用策略 目标对象实现接口时采用 目标对象未实现接口时采用
  • 默认为 JDK ,使用时可以进行指定
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容