美文网首页
JDK和CGLib动态代理

JDK和CGLib动态代理

作者: 压根儿没快乐过 | 来源:发表于2017-09-25 09:28 被阅读0次

    [TOC]

    一:JDK动态代理

    1.创建接口

    public interface Subject {
        public void rent();
    
        public void hello(String str);
    }
    

    2.创建接口实现类

    public class RealSubject implements Subject {
    
        public void rent() {
            System.out.println("I want to rent my house");
        }
    
        public void hello(String str) {
            System.out.println("hello: " + str);
        }
    }
    

    3.创建动态代理类

    public class DynamicProxy implements InvocationHandler {
    
        private Object subject;
    
        //构造方法,给我们要代理的真实对象赋初值
        public DynamicProxy(Object subject) {
            this.subject = subject;
        }
    
        //实现接口的方法
        public Object invoke(Object object, Method method, Object[] args) throws Throwable {
            //在代理真实对象前我们可以添加一些自己的操作
            System.out.println("before rent house");
    
            System.out.println("Method:" + method);
    
            //当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用
            method.invoke(subject, args);
    
            //在代理真实对象后我们也可以添加一些自己的操作
            System.out.println("after rent house");
    
            return null;
        }
    
    }
    
    }
    

    4.创建客户端

    public class Client {
    
        public static void main(String[] args) {
            //我们要代理的真实对象
            Subject realSubject = new RealSubject();
    
            //我们要代理哪个真实对象,就将该对象传进去,最后是通过该真实对象来调用其方法的
            InvocationHandler handler = new DynamicProxy(realSubject);
            
            /*
             * 通过Proxy的newProxyInstance方法来创建我们的代理对象,我们来看看其三个参数
             * 第一个参数 handler.getClass().getClassLoader() ,我们这里使用handler这个类的ClassLoader对象来加载我们的代理对象
             * 第二个参数realSubject.getClass().getInterfaces(),我们这里为代理对象提供的接口是真实对象所实行的接口,表示我要代理的是该真实对象,这样我就能调用这组接口中的方法了
             * 第三个参数handler, 我们这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上
             */
            Subject subject = (Subject) Proxy.newProxyInstance(handler.getClass().getClassLoader(), realSubject .getClass().getInterfaces(), handler);
    
            System.out.println(subject.getClass().getName());
    
            //调用第一个方法
            subject.rent();
    
            //调用第二个方法
            subject.hello("world");
    
        }
    
    }
    

    二:CLIiB动态代理

    JDK实现动态代理需要实现类通过接口定义业务方法,对于没有接口的类,如何实现动态代理呢,这就需要CGLib了。CGLib采用了非常底层的字节码技术,其原理是通过字节码技术为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。

    1.创建对象

    这是一个需要被代理的类,也就是父类,通过字节码技术创建这个类的子类,实现动态代理。

    public class SayHello {
        public void say(){
            System.out.println("hello everyone");
        }
    }
    

    2.创建动态代理类

    该类实现了创建子类的方法与代理的方法。getProxy(SuperClass.class)方法通过入参即父类的字节码,通过扩展父类的class来创建代理对象。intercept()方法拦截所有目标类方法的调用,obj表示目标类的实例,method为目标类方法的反射对象,args为方法的动态入参,proxy为代理类实例。proxy.invokeSuper(obj, args)通过代理类调用父类中的方法。

    public class CglibProxy implements MethodInterceptor {
    
        private Enhancer enhancer = new Enhancer();
    
        public Object getProxy(Class clazz){
            //设置需要创建子类的类
            enhancer.setSuperclass(clazz);
            enhancer.setCallback(this);
            //通过字节码技术动态创建子类实例
            return enhancer.create();
        }
    
        //实现MethodInterceptor接口方法
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
    
            System.out.println("前置代理");
    
            //通过代理类调用父类中的方法
            Object result = proxy.invokeSuper(obj, args);
    
            System.out.println("后置代理");
    
            return result;
        }
    }
    

    3.客户端

    public class DoCGLib {
        public static void main(String[] args) {
            CglibProxy proxy = new CglibProxy();
            //通过生成子类的方式创建代理类
            SayHello proxyImp = (SayHello)proxy.getProxy(SayHello.class);
            proxyImp.say();
        }
    }
    

    三:对比

    CGLib创建的动态代理对象性能比JDK创建的动态代理对象的性能高不少,但是CGLib在创建代理对象时所花费的时间却比JDK多得多,所以对于单例的对象,因为无需频繁创建对象,用CGLib合适,反之,使用JDK方式要更为合适一些。同时,由于CGLib由于是采用动态创建子类的方法,对于final方法,无法进行代理。

    四:参考

    https://mp.weixin.qq.com/s?__biz=MzIwMTY0NDU3Nw==&mid=2651934527&idx=1&sn=46b53438fdd684159025311fcdb07838&chksm=8d0f2071ba78a9670b4091675582434e0aacfe5245f7692ef4269c8e929b3489b52167f9e92f&mpshare=1&scene=23&srcid=0913iapuMSmotjiEpG47zrPR#rd

    http://blog.csdn.net/yakoo5/article/details/9099133/

    相关文章

      网友评论

          本文标题:JDK和CGLib动态代理

          本文链接:https://www.haomeiwen.com/subject/odsrextx.html