美文网首页Android设计模式
设计模式笔记三代理模式

设计模式笔记三代理模式

作者: summer_lz | 来源:发表于2017-03-11 15:07 被阅读6次

每日一文

捭阖之道,以阴阳试之,故与阳言者依崇高,与阴言者依卑小。以下求小,以高求大。由此言之,无所不出,无所不入,无所不可。可以说人,可以说家,可以说国,可以说天下。
关于开放和封闭的规律都要从有阳两方面来试验。因此,给从阳的方面来游说的人以崇高的待遇,而给从阴的方面来游说的人以卑下的待遇。用卑下的来求索微小,以崇高来求索博大。由此看来,没有什么不能出去,没有什么不能进来,没有什么办不成的。用这个道理,可以说服人,可以说服家,可以说服国,可以说服天下。

代理模式


作用

原有类的行为基础上,加入一些多出的行为,甚至完全替换原有的行为。


代理模式中的角色:

  • 抽象对象:声明与丑代理对象一样的街口,使用目标对象的地方完全可以由代理对象替代。
  • 目标对象:被代理的对象
  • 代理对象:代理对象内部含有目标对象的引用,从而可以在任何时候操作目标对象;代理对象提供一个与目标对象相同的接口,以便可以在任何时候替代目标对象。代理对象通常在客户端调用传递给目标对象之前或之后,执行某个操作,而不是单纯地将调用传递给目标对象。

两种形式:

  • 静态代理:

指程序员创建好代理类,编译时直接生成代理类的字节码文件。

特点:

代理类和委托类实现了相同的接口,代理类通过委托类实现了相同的方法。这样就出现了大量的代码重复。而且代理类只能为特定的接口(Service)服务。

示例:
  1. 统一的抽象接口:
package com.bankht.Proxy;
public abstract class AbstractObject {
      // 操作
      public abstract void operation();
}
  1. 目标类:
package com.bankht.Proxy;
public class RealObject extends AbstractObject {
      @Override
      public void operation() {
        // 一些操作
        System.out.println("一些操作");
      }
 }
  1. 静态代理类:对目标类的封装
package com.bankht.Proxy;
public class ProxyObject extends AbstractObject {
      RealObject realObject = new RealObject();

      @Override
      public void operation() {
        // 调用目标对象之前可以做相关操作
        System.out.println("before");
        realObject.operation();
        // 调用目标对象之后可以做相关操作
        System.out.println("after");
      }
}
  1. 客户端调用:
package com.bankht.Proxy;
public class Client {
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        AbstractObject obj = new ProxyObject();
        obj.operation();
      }
}
  • 动态代理:

代理类可代理一系列类的特定方法,代理类需要实现InvocationHandler接口。

特点:

如果需要委托类处理某一业务,那么我们就可以先在代理类中,对客户的权限、各类信息先做判断,如果不满足某一特定条件,则将其拦截下来,不让其代理。

示例:
  1. 代理的接口:代码引用需要空行
    package com.example.patternproxy;

        import java.util.Date;
        
        /**
         * Created on 2017/3/11.
         * Desc:代理的接口
         * Author:Eric.w
         */
        
        public interface DateSerice {
        
            String queryDate();
        
            int cauOld(Date startdate);
        }
    
  2. 被代理的类:

     package com.example.patternproxy;
    
     import android.util.Log;
    
     import java.text.SimpleDateFormat;
     import java.util.Date;
    
     /**
     * Created on 2017/3/11.
     * Desc:
     * Author:Eric.w
     */
    
     public class DateServiceImp implements DateSerice {
    
        SimpleDateFormat myFmt = new SimpleDateFormat("yyyy年MM月dd日 HH时mm分ss秒");
    
        @Override
        public String queryDate() {
            Date date = new Date();
            Log.e("pattern", "时间是:" + myFmt.format(date));
            return myFmt.format(date);
        }
    
        @Override
        public int cauOld(Date startdate) {
            return new Date().getYear() - startdate.getYear();
        }
    
        public String ownMethod() {
            Log.e("pattern", "ownMethod:");
            return "admin";
        }
     }
    
  3. 代理类

  • 动态代理有一个强制性要求,就是被代理的类必须实现了某一个接口,或者本身就是接口,就像我们的Connection。
    道理其实很简单,这是因为动态代理生成的代理类是继承Proxy类的,并且会实现被你传入newProxyInstance方法的所有接口,所以我们可以将生成的代理强转为任意一个代理的接口或者Proxy去使用,但是Proxy里面几乎全是静态方法,没有实例方法,所以转换成Proxy意义不大,几乎没什么用。假设我们的类没有实现任何接口,那么就意味着你只能将生成的代理类转换成Proxy,那么就算生成了,其实也没什么用,而且就算你传入了接口,可以强转,你也用不了这个没有实现你传入接口的这个类的方法。
    你可能会说,假设有个接口A,那我将接口A传给newProxyInstance方法,并代理一个没实现接口A的类B,但类B与接口A有一样的方法可以吗?
    答案是可以的,并且JDK的动态代理只认你传入的接口,只要你传入,你就可以强转成这个接口,这个一会解释,但是你无法在invoke方法里调用method.invoke方法,也就是说,你只能全部替换A接口的方法,而不能使用类B中原有与接口A方法描述相同的方法,这是因为invoke中传入的Method的class信息是接口A,而类B因为没实现接口A,所以无法执行传入的Method,会抛出非法参数异常。
  • 上面我们运行就会发现接口的方法全部都只能输出一个很2的字符串了。如果是要继续使用TestClass的方法也不是不行,只要你确认你传入的类包括了所有你传入的接口的方法,只是没实现这些接口而已,那么你可以在invoke中这样使用。
public Object invoke(Object proxy, Method method, Object[] args) throws    Throwable {
          System.out.println("before");
          Method sourceMethod = source.getClass().getDeclaredMethod(method.getName(),  method.getParameterTypes());
        sourceMethod.setAccessible(true);
          Object result = sourceMethod.invoke(source, args);
          System.out.println("after");
          return result;
    }

这就与你实现接口的表现行为一致了,但是我们本来就只需要一句method.invoke就可以了,就因为没实现接口就要多写两行,所以这种突破JDK动态代理必须实现接口的行为就有点画蛇添足了。因为你本来就实现了该接口的方法,只是差了那一句implements而已。
上面写这个例子只是为了解释LZ当初的疑惑,因为LZ曾一度认为不实现接口就不能使用动态代理

  1. 代码
    import android.util.Log;
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    /**
     * Created on 2017/3/11.
     * Desc:
     * Author:Eric.w
     */
    
    public class DateProxy implements InvocationHandler {
    
        private Object target;
    
        public DateProxy(Object target) {
            this.target = target;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            if (DateSerice.class.isAssignableFrom(proxy.getClass()) &&
                    method.getName().equals("queryDate")) {
                //调用代理
                Log.e("pattern", "方法来自代理!");
                return null;
            }
            //不调用代理
            return method.invoke(proxy, args);
        }
    
        /**
         * 返回被代理的类实例
         *
         * @return
         */
        public Object getProxyInstance() {
            return Proxy.newProxyInstance(target.getClass().getClassLoader()
                    , target.getClass().getInterfaces()
                    , this);
        }
    }

讲解动态代理的原理

相关文章

  • 设计模式之代理模式

    title: 设计模式之代理模式date: 2018-08-15 20:25:42tags: 设计模式 笔记 Ja...

  • Proxy 代理模式

    设计原则学习笔记 设计模式学习笔记 作用 为对象提供增强作用 类图 图为静态代理。另有动态代理模式,由于类图不具...

  • 前端设计模式

    JS设计模式一:工厂模式jS设计模式二:单例模式JS设计模式三:模块模式JS设计模式四:代理模式JS设计模式五:职...

  • 手撸golang 结构型设计模式 代理模式

    手撸golang 结构型设计模式 代理模式 缘起 最近复习设计模式拜读谭勇德的<<设计模式就该这样学>>本系列笔记...

  • 《JS设计模式》读书笔记(一)

    标签:JS 设计模式 《JS设计模式》读书笔记(二) 《JS设计模式》读书笔记(三) 《JS设计模式》读书笔记(四...

  • 26 Java设计模式系列-代理模式

    代理模式 代理模式是非常常见的设计模式之一,写个笔记,记录一下我的学习过程和心得。 首先了解一些代理模式的定义。 ...

  • 《JS设计模式》读书笔记(二)

    标签:JS 设计模式 读书笔记链接: 《JS设计模式》读书笔记(一) 《JS设计模式》读书笔记(三) 《JS设计模...

  • 实习第七十四天(设计模式)

    参考 《JS设计模式》读书笔记(二) 《JS设计模式》读书笔记(三) 《JS设计模式》读书笔记(四) 《JS设计模...

  • Proxy代理者模式(一)

    摘要 本篇笔记针对Java设计模式中最难理解的代理者模式进行讲解,从静态代理、动态代理,及Java相关代理类的应用...

  • 设计模式系列笔记-代理模式

    写在前面:本系列文章内容为《JavaScript设计模式与开发实践》一书学习笔记,感谢作者曾探 代理模式 代理模式...

网友评论

    本文标题:设计模式笔记三代理模式

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