死磕cglib系列之一 cglib简介与callback解析「建议收藏」

死磕cglib系列之一 cglib简介与callback解析「建议收藏」简介cglib是一套java动态代理实现框架,cglib被应用到springapp,hibernate等高级业务框架,spring事务在业务实现类未实现接口的情况下也会使用该技术。实际上,cglib基于继承实现,这也就意味着final,private相关的method无法被代理。基于asm框架对class字节码编辑改动,从而达到动态代理的目的,总之,被代理类没有实现接口的情况下cglib为首…

大家好,又见面了,我是你们的朋友全栈君。

简介

cglib是一套java动态代理实现框架,cglib被应用到spring app,hibernate等高级业务框架,spring事务在业务实现类未实现接口的情况下也会使用该技术。

实际上,cglib基于继承实现,这也就意味着final,private相关的method无法被代理。基于asm框架对class字节码编辑改动,从而达到动态代理的目的,总之,被代理类没有实现接口的情况下cglib为首选

引用官网介绍的一段话:

Byte Code Generation Library is high level API to generate and transform JAVA byte code. It is used by AOP, testing, data access frameworks to generate dynamic proxy objects and intercept field access.

github地址:https://github.com/cglib/cglib

cglib作者博客:http://mydailyjava.blogspot.no/2013/11/cglib-missing-manual.html

以下案例在我的github中可以clone到,分支为howtocglib

github路径:https://github.com/zhang6622056/cglib

测试用例:cglib-sample/src/main/test/callback/Callbacktest

实现动态代理的几种方案

  • javasisit
  • jdk 动态代理
  • cglib
  • asm(cglib 内部也是用asm更改其字节码)

callbacks简介

这里的callback可以认为是cglib用于生成字节码的实现手段,cglib一共实现了6种callback,用于对代理类目标进行不同手段的代理,非常灵活,分别为:

  • FixedValue
  • InvocationHandler
  • LazyLoader
  • MethodInterceptor
  • Dispatcher
  • NoOp

编写实现案例

callbacks代码编写


Dispatcher

实现Dispatcher接口,要求实现loadObject方法,返回期望的代理类。值的一提的是,loadobject方法在每次调用被拦截方法的时候都会被调用一次

/**** * 与lazy不同的是,每一次调用代理方法的时候,都会调用一次Dispatcher的loadObject获取对象 * 而lazy则会缓存下来。 */
public class DispatcherCallBack implements Dispatcher { 
   
    public Object loadObject() throws Exception { 
   
        CallbackBean callbackBean = new CallbackBean();
        return callbackBean;
    }
}

FixedValue

实现FixedValue接口,该callback同样要求实现一个loadobject方法,只不过需要注意的是该loadobject方法相同与重写了被代理类的相应方法,因为在被代理之后,FixedValue callback只会调用loadobject,而不会再调用代理目标类的相应方法!

/***** * 该callback相当于重写了相应的函数实现。并不会调用原函数 */
public class FixValueCallback implements FixedValue { 
   


    /***** * 被代理方法的指定函数将会无条件的返回改object,动态的变更返回值 * @return * @throws Exception */
    public Object loadObject() throws Exception { 
   
        System.out.println("this is fixvalue callback ..... overwrite the code....");
        return true;
    }
}

InvocationHandler

需要实现InvocationHandler接口,实现invoke对象,该拦截传入了proxy对象,用于自定义实现,与MethodInterceptor相似,慎用method的invoke方法。切忌不要造成循环调用

public class InvocationHandlerCallback implements InvocationHandler { 
   
    /***** * invocationHandler的invoke方法传入的method和proxy都是代理本身对象 * 切忌重复调用,会循环调用 * @param proxy 代理类本身 * @param method 代理类内部的方法 * @param args 参数 * @return * @throws Throwable */
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
   
        System.out.println("invocationHandlerCallback Before....");
        method.invoke(proxy.getClass().getSuperclass().newInstance(),args);
        //会无限循环
        //method.invoke(proxy,args);
        System.out.println("invocationHandlerCallback after....");
        return null;
    }
}

LazyLoader

实现LazyLoader的loadObject方法,返回对象实例,该实例只有第一次调用的时候进行初始化,之后不再重新调用,proxy类初始化时进行了成员的赋值,之后使用该成员进行调用父类方法

/**** * * 延迟加载初始化 * 类似于spring prototype的singleton ,在第一次调用的时候进行初始化,并且将此实例存储起来,之后都将返回改实例 * 可参考资料: * https://shensy.iteye.com/blog/1881277 */
public class LazyLoaderCallback implements LazyLoader { 
   
    public Object loadObject() throws Exception { 
   
        CallbackBean callbackBean = new CallbackBean();
        return callbackBean;
    }
}

MethodInterceptor

实现MethodInterceptor的intercept,实现被代理对象的逻辑植入。也是最常用的callback

/**** * 生成代理类字节码。对方法进行拦截调用 */
public class MethodInterceptorCallback implements MethodInterceptor { 
   
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { 
   
        System.out.println("before invoke........");
        proxy.invokeSuper(obj,args);
        System.out.println("after invoke.........");
        return null;
    }
}

NoOp

通过接口声明了一个单例对象,该代理不对被代理类执行任何操作

/** * Methods using this {@link Enhancer} callback will delegate directly to the * default (super) implementation in the base class. */
public interface NoOp extends Callback
{ 
   
    /** * A thread-safe singleton instance of the <code>NoOp</code> callback. */
    public static final NoOp INSTANCE = new NoOp() { 
    };
}

被代理类code


public class CallbackBean { 
   

    public void methodForDispatcher(){ 
   
        System.out.println("methodForDispatcher...");
    }

    public void methodForFixValue(){ 
   
        System.out.println("methodForFixValue...");
    }

    public void methodForInvocationHandler(){ 
   
        System.out.println("methodForInvocationHandler...");
    }

    public void methodForLazy(){ 
   
        System.out.println("methodForLazy...");
    }

    public void methodForInterceptor(){ 
   
        System.out.println("methodForInterceptor...");
    }

    public void methodForNoop(){ 
   
        System.out.println("methodForNoop...");
    }
}

编写Enhancer调用


import callbacks.*;
import net.sf.cglib.proxy.Callback;
import net.sf.cglib.proxy.CallbackFilter;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.NoOp;
import org.junit.Test;
import proxybean.CallbackBean;
import java.lang.reflect.Method;

public class CallbackTest { 
   

    @Test
    public void testCallback(){ 
   
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(CallbackBean.class);
        enhancer.setCallbacks(initCallBacks());
        enhancer.setCallbackFilter(initCallbackFilter());
        CallbackBean callbackBean = (CallbackBean) enhancer.create();

        callbackBean.methodForNoop();
        callbackBean.methodForInterceptor();
        callbackBean.methodForLazy();
        callbackBean.methodForDispatcher();
        callbackBean.methodForInvocationHandler();
        callbackBean.methodForFixValue();
    }





    /**** * 初始化callback拦截数组 * @return */
    private static final Callback[] initCallBacks(){ 
   
        MethodInterceptorCallback methodInterceptorCallback = new MethodInterceptorCallback();
        LazyLoaderCallback lazyLoaderCallback = new LazyLoaderCallback();
        InvocationHandlerCallback invocationHandlerCallback = new InvocationHandlerCallback();
        FixValueCallback fixValueCallback = new FixValueCallback();
        DispatcherCallBack dispatcherCallBack = new DispatcherCallBack();
        Callback[] callbacks = new Callback[]{ 
   NoOp.INSTANCE,methodInterceptorCallback,lazyLoaderCallback,dispatcherCallBack,invocationHandlerCallback,fixValueCallback};
        return callbacks;
    }



    private static final CallbackFilter initCallbackFilter(){ 
   
        return new CallbackFilter() { 
   
            public int accept(Method method) { 
   
                if (method.getName().equals("methodForNoop")){ 
   
                    return 0;
                }
                if (method.getName().equals("methodForInterceptor")){ 
   
                    return 1;
                }
                if (method.getName().equals("methodForLazy")){ 
   
                    return 2;
                }
                if (method.getName().equals("methodForDispatcher")){ 
   
                    return 3;
                }
                if (method.getName().equals("methodForInvocationHandler")){ 
   
                    return 4;
                }
                if (method.getName().equals("methodForFixValue")){ 
   
                    return 5;
                }
                return 0;
            }
        };

    }
}

proxy class反编译分析(请看关键注释)

Dispatcher

public final void methodForDispatcher() { 
   
    Dispatcher var10000 = this.CGLIB$CALLBACK_3;
    if (var10000 == null) { 
   
        CGLIB$BIND_CALLBACKS(this);
        var10000 = this.CGLIB$CALLBACK_3;
    }
	//每次都调用一次loadObject,获取对象,并调用对象的相应方法
    //这样的实现,相当于loadObject可以很灵活的返回相应的实现类或者子类
    ((CallbackBean)var10000.loadObject()).methodForDispatcher();
}

FixedValue

public final void methodForFixValue() { 
   
    FixedValue var10000 = this.CGLIB$CALLBACK_5;
    if (var10000 == null) { 
   
        CGLIB$BIND_CALLBACKS(this);
        var10000 = this.CGLIB$CALLBACK_5;
    }
	//直接调用了FixValue callback的loadObject,相当于重写逻辑
    var10000.loadObject();
}

InvocationHandler

public final void methodForInvocationHandler() { 
   
    try { 
   
        InvocationHandler var10000 = this.CGLIB$CALLBACK_4;
        if (var10000 == null) { 
   
            CGLIB$BIND_CALLBACKS(this);
            var10000 = this.CGLIB$CALLBACK_4;
        }
		//调用InvocationHandler相应的invoke实现
        var10000.invoke(this, CGLIB$methodForInvocationHandler$4, new Object[0]);
    } catch (Error | RuntimeException var1) { 
   
        throw var1;
    } catch (Throwable var2) { 
   
        throw new UndeclaredThrowableException(var2);
    }
}

LazyLoader

public final void methodForLazy() { 
   
	//CGLIB$LOAD_PRIVATE_2在proxy初始化的时候初始化一次,并维护了该成员
	//与Dispatcher不同,每次调用都是运用proxy成员变量进行调用
    ((CallbackBean)this.CGLIB$LOAD_PRIVATE_2()).methodForLazy();
}

MethodInterceptor

public final void methodForInterceptor() { 
   
    MethodInterceptor var10000 = this.CGLIB$CALLBACK_1;
    if (var10000 == null) { 
   
        CGLIB$BIND_CALLBACKS(this);
        var10000 = this.CGLIB$CALLBACK_1;
    }

    if (var10000 != null) { 
   
        //调用callback的intercept方法,用于实现拦截
        var10000.intercept(this, CGLIB$methodForInterceptor$1$Method, CGLIB$emptyArgs, CGLIB$methodForInterceptor$1$Proxy);
    } else { 
   
        super.methodForInterceptor();
    }
}

NoOp


此处不做NoOp的讨论,理解为调用原类方法即可

运行结果

在这里插入图片描述

反编译遇到问题

如jd-gui反编译遇到错误,请尝试直接将class文件拖进idea

总结

以上即为本篇的全部内容,主要简单介绍了cglib与几种callback的使用。比较简单、基础。

当然,cglib不只是这些内容。在实现的生成proxy的同时,作者抽象了很多东西,这包括

  • 缓存机制

  • 生成策略

  • 弱引用WeakHashMap,WeakReference

  • LoadingCache搭配FutureTask线程竞争模型

  • 对反射的封装

  • 对asm的封装

  • 类加载的ClassLoader

  • 灵活的Function设计

总之,先会用,我们在逐一深入,下一篇正式开始撸源码,敬请期待。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/139892.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Mac os 安装Python Pycharm 配置环境「建议收藏」

    Mac os 安装Python Pycharm 配置环境「建议收藏」  主要就是这三个库的安装   importrequestsfrombs4importBeautifulSoupimporttime我是PYthon小白,自己把程序运行出来在环境配置走了不少弯路。因为我还安装了一台Windows环境,中间交叉做了其他一些事情,所以思路没有那么清晰。但是刚刚终于成功抓了数据。代码和程序运行成功截图放在最后。先说说环境配置,我会尽量回忆。我安装的是…

    2022年8月28日
    4
  • 递归迭代动态规划「建议收藏」

    递归迭代动态规划「建议收藏」一、定义递归:程序调用自身,从顶部将问题分解,其问题与其子问题是同一概念。通过解决掉所有分解出来的小问题,来解决整个问题。迭代:利用变量的原值推算出变量的下一个值。递归中一定有迭代,但是迭代中不一定有递归。动态规划:通常与递归相反,其从底部开始解决问题。将所有小问题解决掉,进而解决的整个问题。为了节约重复求相同子问题的时间,引入一个数组,把所有子问题的解存于该数组中,动态规划算法是空间换时间的算法。动态规划可以递归地实现,也可以非递归(循环的方法)地实现。运行速度:动态规划>迭代&gt

    2025年7月1日
    3
  • Linux服务器配置(详细版)[通俗易懂]

    本来就很麻烦,如果错了一个很有可能从新再来,非常麻烦所以要细心Linux命令1.进入文件夹cd文件夹名称2.复制文件夹cp文件名指定目录copy文件到指定目录例如cp文件名称文件路径/var/www例子cp/root/gqxnb/gqx/var/www3.查看文件夹目录lsattr[-adRvV][文件或目录…]选…

    2022年4月14日
    47
  • Linux offsetof宏定义

    Linux offsetof宏定义#include&lt;stddef.h&gt;size_t offsetof(type, member) #define offsetof(TYPE, MEMBER)        \           ((size_t)&amp;((TYPE*)0)-&gt;MEMBER) Themacroreturntheoffsetofthe…

    2022年8月22日
    8
  • Lighttpd 插件mod_h264 streaming (mp4)安装

    Lighttpd 插件mod_h264 streaming (mp4)安装

    2022年3月5日
    38
  • Python优秀开源项目Rich源码解析

    Python优秀开源项目Rich源码解析这篇文章对优秀的开源项目Rich的源码进行解析,OMG,盘他。为什么建议阅读源码,有两个原因,第一,单纯学语言很难在实践中灵活应用,通过阅读源码可以看到每个知识点的运用场景,印象会更深,以后写代码的时

    2022年7月5日
    26

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号