zoukankan      html  css  js  c++  java
  • java动态代理和cglib

    1.代理类可以分为两种。 
    静态代理:由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。 
    动态代理:在程序运行时,运用反射机制动态创建而成。 

    2.JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK代理,这就要使用cglib动态代理了。 

    3.Cglib动态代理 
    JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。 

    4.

    JDK动态代理中包含一个类和一个接口: 
    InvocationHandler接口: 
    public interface InvocationHandler { 
    public Object invoke(Object proxy,Method method,Object[] args) throws Throwable; 

    参数说明: 
    Object proxy:指被代理的对象。 
    Method method:要调用的方法 
    Object[] args:方法调用时所需要的参数 

    可以将InvocationHandler接口的子类想象成一个代理的最终操作类,替换掉ProxySubject。 

    Proxy类: 
    Proxy类是专门完成代理的操作类,可以通过此类为一个或多个接口动态地生成实现类,此类提供了如下的操作方法: 
    public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, 
    InvocationHandler h) 
                                   throws IllegalArgumentException 
    参数说明: 
    ClassLoader loader:类加载器 
    Class<?>[] interfaces:得到全部的接口 
    InvocationHandler h:得到InvocationHandler接口的子类实例 

    Ps:类加载器 
    在Proxy类中的newProxyInstance()方法中需要一个ClassLoader类的实例,ClassLoader实际上对应的是类加载器,在Java中主要有一下三种类加载器; 
    Booststrap ClassLoader:此加载器采用C++编写,一般开发中是看不到的; 
    Extendsion ClassLoader:用来进行扩展类的加载,一般对应的是jrelibext目录中的类; 
    AppClassLoader:(默认)加载classpath指定的类,是最常使用的是一种加载器。 

    动态代理 
    与静态代理类对照的是动态代理类,动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。 

    例如:

    public interface OderFacade {  
        public void addOrder();  
    } 
    public class OrderFacadeImpl implements OrderFacade {  
      
        @Override  
        public void addOrder() {  
            System.out.println("增加订单");  
        }  
      
    }  
    /** 
     * JDK动态代理代理类 
     *  
     */  
    public class OrderFacadeProxy implements InvocationHandler {  
        private Object target;  
        /** 
         * 绑定委托对象并返回一个代理类 
         * @param target 
         * @return 
         */  
        public Object bind(Object target) {  
            this.target = target;  
            //取得代理对象  
            return Proxy.newProxyInstance(target.getClass().getClassLoader(),  
                    target.getClass().getInterfaces(), this);   //要绑定接口(这是一个缺陷,cglib弥补了这一缺陷)  
        }  
      
        @Override  
        /** 
         * 调用方法 
         */  
        public Object invoke(Object proxy, Method method, Object[] args)  
                throws Throwable {  
            Object result=null;  
            System.out.println("事物开始");  
            //执行方法  
            result=method.invoke(target, args);  
            System.out.println("事物结束");  
            return result;  
        }  
      
    }  
    public class TestProxy {  
      
        public static void main(String[] args) {  
            OrderFacadeProxy proxy = new OrderFacadeProxy();  
            OrderFacade bookProxy = (OrderFacade) proxy.bind(new OrderFacadeImpl());  
             orderProxy.addOrder();  
        }  
      
    } 

    CgLib运用的代码实例:

    要被代理的类:

    public class SayHello {
        
        public void sayHello(String words){
            System.out.println(words);
        }
    
    }

    要引入Spring-core包,代理类:

    public class SayHelloProxy implements MethodInterceptor{
        
        private Enhancer enhancer = new Enhancer();
        
        public Object getProxy(Class clazz){
              enhancer.setSuperclass(clazz);
              enhancer.setCallback(this);
    
              return enhancer.create();
             }
    
        @Override
        public Object intercept(Object object, Method method, Object[] args,
                MethodProxy methodProxy) throws Throwable {
            System.out.println("调用之前处理");
            methodProxy.invokeSuper(object, args);
            System.out.println("调用之后处理");
            return null;
        }
        
    }

    测试类:

    public class TestCgLibProxy {
    
        public static void main(String[] args) {
            
            SayHelloProxy proxy = new SayHelloProxy();
            SayHello sayHello = (SayHello) proxy.getProxy(SayHello.class);
            sayHello.sayHello("Hello World!");
        }
    
    }

    运行结果:

    调用之前处理
    Hello World!
    调用之后处理

    总结:

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

    2.JDK动态代理只能代理集成了接口的类,CGLib能代理无限制。

  • 相关阅读:
    LeetCode 32.使数组唯一的最小增量
    LeetCode 31. 最小的k个数 快速排序+堆排序+二叉搜索树
    LeetCode 30. 最长回文串
    LeetCode 29. 矩形重叠 反向思维
    LeetCode 28. 拼写单词 HashMap赋值给另一个HashMap
    LeetCode 27. 字符串压缩
    Java SSM Spring+Spring MVC+Mybatis整合
    LeetCode 26.岛屿的最大面积 DFS深度遍历问题
    LeetCode 25.最长上升子序列 动态规划
    LeetCode 24.找出数组中出现次数大于二分之一数组长度的数
  • 原文地址:https://www.cnblogs.com/suncj/p/4700137.html
Copyright © 2011-2022 走看看