zoukankan      html  css  js  c++  java
  • CGLIB代理基础

      本文意在讲解CGLIB的基础使用及基本原理。

    一、CGLIB的基本原理:

      依赖ASM字节码工具,通过动态生成实现接口或继承类的类字节码,实现动态代理。

      针对接口,生成实现接口的类,即implements方式;针对类,生成继承父类的类,即extends方式。

    二、为什么使用CGLIB?

      JDK的动态代理只能基于接口,有时候我们想基于类生成动态代理,这个时候CGLIB是一个选择。

    没什么场景下是必须使用CGLIB生成类代理的(个人观点),如果有,可能是代码简洁,某些情况下性能较好。

      CGLIB基于类生成动态代理需要注意?(CGLIB生成的代理是继承类的)

      1.  final声明的类是不能被代理的;

      2.  类中的private,final方法不能被代理,static方法不生成代理方法。

    二、使用方法:

      基础示例代码:

    public interface UserInterface {
        boolean login(int userid);
    }
    public class UserInterfaceImpl implements UserInterface {
        public boolean login(int userid) {
            System.out.println("do Login!");
            return false;
        }
    }

      1. 代理接口:

    public class CGlibProxy implements net.sf.cglib.proxy.InvocationHandler{ //这里的InvocationHandler是cglib包中的
        private UserInterface ref;
        public CGlibProxy(UserInterface ref){
            this.ref = ref;
        }
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("before");
            method.invoke(ref, args);
            System.out.println("after");
            return null;
        }
    }
    public class Test {
        public static void main(String[] args) throws IOException {
            Enhancer en = new Enhancer();
            en.setSuperclass(UserInterface.class);
            en.setCallback(new CGlibProxy(new UserInterfaceImpl()));
            UserInterface interfaced = (UserInterface) en.create();
            interfaced.login(1);
        }
    }

      代理接口和JDK的使用方法基本没啥区别,传入被代理实例对象,调用实例的对象的method方法。

      所以如果是代理接口,完全没必要使用CGLIB。

      2. 代理类:

    public class CGlibProxy implements MethodInterceptor{
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            System.out.println("before");
            Object o = proxy.invokeSuper(obj, args);
            System.out.println("after");
            return o;
        }
    }
    public class Test {
        public static void main(String[] args) throws IOException {
            Enhancer en = new Enhancer();
            en.setSuperclass(UserInterfaceImpl.class);
            en.setCallback(new CGlibProxy());
            UserInterface interfaced = (UserInterface) en.create();
            interfaced.login(1);
        }
    }

      3. Callback接口

       Callback即代理方法,上述示例中MethodInterceptor就是Callback的子接口。Callback定义一个空接口,可以方便扩展。

       Enhancer中有两个设置Callback的方法: setCallback(Callback callback), setCallbacks(Callback[] callbacks)。

       你可能定义多个Callback,然后定义不同的代理行为,如下:

    public class LoginProxy implements MethodInterceptor{
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            System.out.println("before login");
            proxy.invokeSuper(obj, args);
            System.out.println("after loign");
            return null;
        }
    }
    public class OtherProxy implements MethodInterceptor{
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            System.out.println("before");
            proxy.invokeSuper(obj, args);
            System.out.println("after");
            return null;
        }
    }

        cglib代理只能调用一个代理方法,所以当设置多个Callback时,你还需要指定一个CallbackFilter,通过Method条件指定Callback,

        CallbackFilter接口:

    public interface CallbackFilter {
        int accept(Method method); //返回值为指定的Callback数组的下标索引
    }

      示例: 

    public class Test {
        public static void main(String[] args) throws IOException {
            Enhancer en = new Enhancer();
            en.setSuperclass(UserInterfaceImpl.class);
            en.setCallbacks(new Callback[]{new LoginProxy(),new OtherProxy()}); //callback数组
            en.setCallbackFilter(new CallbackFilter() {
                public int accept(Method method) {
                    if(method.getName().equals("login")){
                        return 0;  //索引为0 , 即  LoginProxy
                    }
                    return 1; // 索引为1, 即 OtherProxy
                }
            });
            UserInterfaceImpl interfaced = (UserInterfaceImpl) en.create();
            interfaced.login(1);
            interfaced.other();
        }
    }

      4. MethodInterceptor:MethodInterceptor接口是Callback的子接口,最常用。

    public interface MethodInterceptorextends Callback { 
      //obj: 代理对象本身,即cglib生成的代理实例
    //method: 被代理对象中的方法
      //args:方法的参数
      //proxy: 存储了代理类,也就是obj
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable; }

              如何调用被代理的目标方法?  

        MethodProxy的 invokeSuper(obj,args) 方法;obj就是 代理对象本身 ,args是对象参数;

        MethodProxy还有一个invoke(obj,args)方法;obj参数是被代理对象,没搞懂这个方法的意图,个人觉得没什么必要。

      5. NamingPolicy,自定义代理类名称,默认实现是DefaultNamingPolicy

    public interface NamingPolicy {
        String getClassName(String prefix, String source, Object key, Predicate names);
    }

      6. GeneratorStrategy,字节码生成策略,默认实现是DefaultGeneratorStrategy

    public interface GeneratorStrategy {
        byte[] generate(ClassGenerator cg) throws Exception;
    }

         你可以重写DefaultGeneratorStrategy中的方法来替换字节码生成器,也可以访问或修改生成的字节码,如下:

    Enhancer en = new Enhancer();
    en.setStrategy(new DefaultGeneratorStrategy(){
           protected byte[] transform(byte[] b) throws Exception {
               return b; //b 是生成的字节码
           }
           protected ClassGenerator transform(ClassGenerator cg) throws Exception {
               return cg; //cg 是字节码生成器
           }
    });

      7. interceptDuringConstruction,设置构造函数中的方法调用是否使用代理方法,默认为true。

             示例:

    public class UserInterfaceImpl implements UserInterface {
        public UserInterfaceImpl(){
            login(1);  //构造函数中调用方法
        }
        public boolean login(int userid) {
            System.out.println("do Login!");
            return false;
        }
    }
    public class Test {
        public static void main(String[] args) throws IOException {
            Enhancer en = new Enhancer();
            en.setInterceptDuringConstruction(true); 
            en.setSuperclass(UserInterfaceImpl.class);
            en.setCallback(new LoginProxy());
            UserInterfaceImpl interfaced = (UserInterfaceImpl) en.create();
            interfaced.login(1);
        }
    }

          运行测试代码,结果如下,即login方法被代理了两次。

       

       en.setInterceptDuringConstruction(false) 时,运行结果如下,即login方法被代理了一次

       

      8. 没有默认构造函数时创建代理的方法:  

    public class UserInterfaceImpl implements UserInterface {
        public UserInterfaceImpl(String param){
        }
        public boolean login(int userid) {
            System.out.println("do Login!");
            return false;
        }
    }
    public class Test {
        public static void main(String[] args) throws IOException {
            Enhancer en = new Enhancer();
            en.setSuperclass(UserInterfaceImpl.class);
            en.setCallback(new LoginProxy());
            UserInterfaceImpl interfaced = (UserInterfaceImpl) en.create(new Class[]{String.class}, new Object[]{"name"}); //创建方法中指定构造函数的参数类型及对应的参数值
            interfaced.login(1);
        }
    }

    三、cglib创建类代理的基本原理

      如果让你实现类代理? ----  难点在哪里?

      1. 创建代理的过程:

       根据各种参数生成缓存的key --> 生成代理类(先缓存获取,缓存没有则用ClassGenerator生成代理类存入缓存) --> 根据代理类构造器生成代理对象实例。

      2. 代理类对象:

          可以设置系统参数cglib.debugLocation,开启代理类存入文件,该参数为文件存储路径。

         示例代码: 

    public class UserInterfaceImpl implements UserInterface {
        public boolean login(int userid) {
            return false;
        }
    }
    public class Test {
        public static void main(String[] args) throws IOException {
            System.setProperty("cglib.debugLocation", "E://test");
            Enhancer en = new Enhancer();
            en.setSuperclass(UserInterfaceImpl.class);
            en.setCallback(new LoginProxy());
            en.create();
        }
    }

         运行后,test目录下会生成一些class文件,找到同包(自己的package)目录,反编译打开(用的luyten,比jdgui好用)

         为了方便看,删除了一些不必要的代码;

    package cglibproxy;
    import java.lang.reflect.*;
    import net.sf.cglib.proxy.*;
    import net.sf.cglib.core.*;
    public class UserInterfaceImpl$$EnhancerByCGLIB$$f2d3e293 extends UserInterfaceImpl implements Factory
    {private MethodInterceptor CGLIB$CALLBACK_0;
        private static Object CGLIB$CALLBACK_FILTER;
    private static final Method CGLIB$login$0$Method; private static final MethodProxy CGLIB$login$0$Proxy;
    private static final Method CGLIB$equals$1$Method; private static final MethodProxy CGLIB$equals$1$Proxy;
    private static final Method CGLIB$toString$2$Method; private static final MethodProxy CGLIB$toString$2$Proxy;
    private static final Method CGLIB$hashCode$3$Method; private static final MethodProxy CGLIB$hashCode$3$Proxy;
    private static final Method CGLIB$clone$4$Method; private static final MethodProxy CGLIB$clone$4$Proxy; final boolean CGLIB$login$0(final int n) { return super.login(n); } public final boolean login(final int n) { MethodInterceptor cglib$CALLBACK_2; MethodInterceptor cglib$CALLBACK_0; if ((cglib$CALLBACK_0 = (cglib$CALLBACK_2 = this.CGLIB$CALLBACK_0)) == null) { CGLIB$BIND_CALLBACKS(this); cglib$CALLBACK_2 = (cglib$CALLBACK_0 = this.CGLIB$CALLBACK_0); } if (cglib$CALLBACK_0 != null) { final Object intercept = cglib$CALLBACK_2.intercept((Object)this, UserInterfaceImpl$$EnhancerByCGLIB$$f2d3e293.CGLIB$login$0$Method, new Object[] { new Integer(n) }, UserInterfaceImpl$$EnhancerByCGLIB$$f2d3e293.CGLIB$login$0$Proxy); return intercept != null && (boolean)intercept; } return super.login(n); } }

        可以从反编译的代理类中看到:

          代理类继承了被代理类,Factory接口是cglib的内部接口,有兴趣的可以去看一下;

          代理类代理了两种方法,一种是被代理类(即我们自定义的方法),一种是Object中的4个方法(toString, equals, hashCode, clone);

          代理类针对每个方法,生成了两个方法(一个代理方法,一个原方法) (为什么这么做?想想MethodProxy.invokeSuper(),这里是一个关键点);

          代理类中的目标方法都使用了final声明,禁止继续被代理;

    以上就是个人总结的cglib的一些基础,水平有限,有问题欢迎评论中指正,谢谢!

    原创文章,转载请注明出处。

        

      

      

         

         

         

         

  • 相关阅读:
    单例模式
    js事件
    oracle_to_excel
    jquery_2
    jquery_1
    4.linux 复制,删除,重命名
    一个tomcat下部署多个springboot项目
    3.centos7 安装oracle
    桥接模式
    组合模式
  • 原文地址:https://www.cnblogs.com/selfchange/p/9828097.html
Copyright © 2011-2022 走看看