死磕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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux内核使用的编程语言_linux内核模块编程

    linux内核使用的编程语言_linux内核模块编程1、内核编程不能访问C库2、内核编程时必须使用GNUC3、内核编程时缺乏像用户空间那样的内存保护机制4、内核编程时浮点数很难使用5、内核只有一个很小的定长堆栈6、由于内核支持异步中断,抢占和SMP,因此必须时刻注意同步和并发7、要考虑可移植性的重要性

    2022年10月8日
    1
  • centos7网络设置ipv4_centos7连接wifi详细步骤

    centos7网络设置ipv4_centos7连接wifi详细步骤系统版本centos7.71、ip配置(配置后局域网内可互ping)同网段内设置,不用网关即可通信1、临时设置#设置接口ens33的地址为192.168.59.27ipaddradd192.168.56.27/24devens33#查看接口ens33地址ipaddrshowens332、永久设置进入目录,/etc/sysconfig/network-scripts,修改文件ifcfg-ens33,修改或添加下列几项BOOTPROTO=”none”#

    2022年5月3日
    190
  • 获取html中光标位置[通俗易懂]

    获取html中光标的位置

    2022年2月26日
    44
  • 【Leetcode】Pascal&#39;s Triangle II

    【Leetcode】Pascal&#39;s Triangle II

    2022年1月1日
    51
  • 2021Vue.js面试题汇总及答案【全网最全 建议收藏】「建议收藏」

    2021Vue.js面试题汇总及答案【全网最全 建议收藏】「建议收藏」文章目录前言一、Vue.js基本问题1.1.Vue响应式原理1.2.Vue.js的特点1.3.Vue.js双向绑定的原理1.4.Vue中如何监控某个属性值的变化?1.5.Vue.js3.0放弃defineProperty,使用Proxy的原因1.6.Vue2中给data中的对象属性添加一个新的属性时会发生什么?如何解决?前言之前博主有分享过Vue学习由浅到深的文章(Vue学习之从入门到神经)现在Vue学的好的话马内真的不必后端差所以今天博主就汇总下有关Vue的相关面试题

    2022年6月12日
    47
  • 图像处理中经常用的一幅图片Lena.jpg

    图像处理中经常用的一幅图片Lena.jpg在数字图像处理中,Lena(Lenna)是一张被广泛使用的标准图片,特别在图像压缩的算法研究中。(为什么用这幅图,是因为这图的各个频段的能量都很丰富:即有低频(光滑的皮肤),也有高频(帽子上的羽毛),很适合来验证各种算法)然而,这张图片背后的故事是颇有意思的,很多人都抱有学究都是呆子的看法,然而Lena对此就是一个有力的驳斥。lena(lenna)是一张于1972年11月…

    2022年6月19日
    78

发表回复

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

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