• java之动态代理


    1.由于静态代理产生的代理类过多,造成代码量急剧增加,所以在实际开发中,通常使用动态代理来实现代理功能。

    2.在java中实现动态代理的方式有很多。Jdk中通过接口来实现动态代理,如果要实现类的动态代理可以使用 cglib。目前也可以通过 javassit 来实现代理。

    3.Jdk的动态代理实现是通过一个类和一个接口来实现的。

    InvocationHandler 是代理实例 的调用处理程序 实现的接口。

    Object invoke(Object proxy,

                  Method method,

                  Object[] args)

                  throws Throwable

    在代理实例上处理方法调用并返回结果。在与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。

    参数: 

    proxy - 在其上调用方法的代理实例

    method - 对应于在代理实例上调用的接口方法的 Method 实例。Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。

    args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null。基本类型的参数被包装在适当基本包装器类(如 java.lang.Integer  java.lang.Boolean)的实例中。

    返回: 

    从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例;否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为 null 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 NullPointerException。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出 ClassCastException

    抛出: 

    Throwable - 从代理实例上的方法调用抛出的异常。该异常的类型必须可以分配到在接口方法的 throws 子句中声明的任一异常类型或未经检查的异常类型 java.lang.RuntimeException  java.lang.Error。如果此方法抛出经过检查的异常,该异常不可分配到在接口方法的 throws 子句中声明的任一异常类型,代理实例的方法调用将抛出包含此方法曾抛出的异常的 UndeclaredThrowableException

    Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。

    public static Object newProxyInstance(ClassLoader loader,

                                          Class<?>[] interfaces,

                                          InvocationHandler h)

                                   throws IllegalArgumentException

    返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。此方法相当于:

         Proxy.getProxyClass(loader, interfaces).

             getConstructor(new Class[] { InvocationHandler.class }).

             newInstance(new Object[] { handler });

    Proxy.newProxyInstance 抛出 IllegalArgumentException,原因与 Proxy.getProxyClass 相同。

    参数: 

    loader - 定义代理类的类加载器

    interfaces - 代理类要实现的接口列表

    h - 指派方法调用的调用处理程序

    返回: 

    一个带有代理类的指定调用处理程序的代理实例,它由指定的类加载器定义,并实现指定的接口

    抛出: 

    IllegalArgumentException - 如果违反传递到 getProxyClass 的参数上的任何限制

    NullPointerException - 如果 interfaces 数组参数或其任何元素为 null,或如果调用处理程序 null

    4.通过动态代理实现租房

    MyInvocationHandler.java

    public class MyInvocationHandler implements InvocationHandler{
        //目标对象 ---真实对象
        private Object target;
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            fare();    
            //调用真实对象 执行其方法
            return method.invoke(target, args);
        }
        private void fare(){
            System.out.println("中介收费");
        }
        public void setTarget(Object target) {
            this.target = target;
        }
    }

    Rent.java

    public interface Rent {
        //租房
        public void rent();
    }

    Host.java

    public class Host implements Rent{
        @Override
        public void rent() {
            System.out.println("========将房屋出租=======");
        }
    }

    Client.java

    public class Client {
        public static void main(String[] args) {
            Host host = new Host();
            MyInvocationHandler handler = new MyInvocationHandler();
            handler.setTarget(host);
            Rent proxy=(Rent)Proxy.newProxyInstance(Client.class.getClassLoader(),
                    host.getClass().getInterfaces(), handler);
            proxy.rent();
        }
    }

    5.封装为一个类

    public class DynamicProxy implements InvocationHandler{
        private Object target;
        public void setTarget(Object target) {
            this.target = target;
        }
        public Object getProxy(){
            return Proxy.newProxyInstance(this.getClass().getClassLoader(),
                    target.getClass().getInterfaces(), 
                    this);
        }
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            before();
            //result方法执行的返回值
            Object result = method.invoke(target, args);
            after();
            return result;
        }
        private void before(){
            System.out.println("在真实方法执行前 添加新的功能");
        }
        private void after(){
            System.out.println("在真实方法执行后 添加新的功能");
        }
    }

    总结:动态代理解决了静态代理的缺点;一个类可以代理多个真实对象。

  • 相关阅读:
    DELPHI中Showmodal与Show的区别
    怎样把一个文件夹里面所有文件的文件名提取出来,放到Excel表格里呢
    Python基础之函数
    Python_Openpyxl
    设计模式
    【转】深入理解递归函数的调用过程
    Java集合的Stack、Queue、Map的遍历
    数据结构-String、char
    leetcode-位运算
    Java web入门
  • 原文地址:https://www.cnblogs.com/Vincent-yuan/p/11267282.html
Copyright © 2020-2023  润新知