• 动态代理的简单实例.


    在上一篇文章《设计模式:Java的代理模式》中讲解了代理模式的静态代理方法。比静态代理更灵活的就是动态代理了。

    Java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类:
     (1)Interface InvocationHandler:该接口中仅定义了一个方法
      public object invoke(Object obj,Method method, Object[] args)
    在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如前文中的request(),args为该方法的参数数组。 这个抽象方法在代理类中动态实现。

    (2)Proxy:该类即为动态代理类,作用类似于前文中的ProxySubject,其中主要包含以下内容:

      a)protected Proxy(InvocationHandler h):构造函数,用于给内部的h赋值。

      b)static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。
      c)static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法)

    所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些 interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。在使用动态代理类时,我们必须实现InvocationHandler接口

    实例:

    一个接口类Subject:

    public interface Subject
    {
    public void request();
    }

    实现Subject接口的实体类RealSubject:

    public class RealSubject implements Subject
    {
    public void request()
    {
    System.out.println("From real subject!");
    }

    }

    动态代理类DynamicSubject:

    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;

    /**
    * 该代理类的内部属性是Object类型,实际使用的时候通过该类的构造方法传递进来一个对象
    * 此外,该类还实现了invoke方法,该方法中的method.invoke其实就是调用被代理对象的将要
    * 执行的方法,方法参数是sub,表示该方法从属于sub,通过动态代理类,我们可以在执行真实对象的方法前后
    * 加入自己的一些额外方法。
    *
    */

    public class DynamicSubject implements InvocationHandler
    {
    private Object sub;

    public DynamicSubject(Object obj)
    {
    this.sub = obj;
    }

    //当外界调用代理类实现的接口方法时,将自动调用此invoke()方法.method参数为外界欲调用的方法(由Proxy生成).
    public Object invoke(Object proxy, Method method, Object[] args)
    throws Throwable
    {
    System.out.println("before calling: " + method);

    //反射调用方法,sub是传入的实体对象,method为外界欲调用的方法.代理功能实现!
    method.invoke(sub, args);

    System.out.println(args == null);

    System.out.println("after calling: " + method);

    return null;
    }
    }


    使用动态代理Client:

    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Proxy;

    public class Client
    {
    public static void main(String[] args)
    {
    RealSubject realSubject = new RealSubject();

    InvocationHandler handler = new DynamicSubject(realSubject);

    Class<?> classType = handler.getClass();

    // 下面的代码一次性生成代理

    //newProxyInstance创建的对象(假设名为:proxy)实现了实体类RealSubject的所有接口(参数2)
    //然后就可以将proxy强制转换为Subject类型.
    Subject subject = (Subject) Proxy.newProxyInstance(classType
    .getClassLoader(), realSubject.getClass().getInterfaces(),
    handler);

    //这里调用了request()方法,实际上就是调用proxy实现的接口的request()方法,该方法会调用DynamicSubject的invoke()方法.
    //并传入request()方法对的method对象(实际上是用反射)和方法参数.
    subject.request();
          
    System.out.println(subject.getClass());

    }
    }

    程序运行结果:

    before calling: public abstract void com.shengsiyuan.dynamicproxy.Subject.request()
    From real subject!
    true
    after calling: public abstract void com.shengsiyuan.dynamicproxy.Subject.request()
    class $Proxy0

     然后我们整合一下,将代理:

    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    import java.util.List;
    import java.util.Vector;

    public class VectorProxy implements InvocationHandler
    {
    private Object proxyObj;

    public VectorProxy(Object obj)
    {
    this.proxyObj = obj;
    }

    /*
    * 该方法生成一个实现了Object的接口的对象
    */
    public static Object factory(Object obj)
    {
    Class<?> classType = obj.getClass();

    return Proxy.newProxyInstance(classType.getClassLoader(),
    classType.getInterfaces(), new VectorProxy(obj));
    }

    public Object invoke(Object proxy, Method method, Object[] args)
    throws Throwable
    {
    //输出参数
    if(null != args)
    {
    for(Object obj : args)
    {
    System.out.println("参数:"+obj);
    }
    }

    Object object = method.invoke(proxyObj, args);

    return object;
    }

    public static void main(String[] args)
    {
    //List是一个接口,Vector实现了List接口
    List v = (List)factory(new Vector());

    System.out.println(v.getClass().getName());

    v.add("New");
    v.add("York");

    System.out.println(v);
    v.remove(0);
    System.out.println(v);

    }
    }

    程序运行结果是:

    $Proxy0
    参数:New
    参数:York
    [New, York]
    参数:0
    [York]



    总结:

    通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject类)也可以动态改变,从而实现了非常灵活的动态代理关系。



    本文为<北京圣思园Java培训教学视频Java.SE>视频笔记

  • 相关阅读:
    将两个数组对比后合并为同一个数组
    invalid reference format: repository name must be lowercase
    Error: too many open files之ulimt
    vim打开文件末尾带有^M
    双层for循环体里,分别跳出外层循环和内层循环
    echarts 多饼图集合多标题
    近1个月订单占比城市TOP6
    javascript 显示日期
    国密SM2,SM4 For Delphi xe 10.3.3
    Datasnap POST 方案
  • 原文地址:https://www.cnblogs.com/myfjd/p/2322480.html
Copyright © 2020-2023  润新知