设计模式笔记三代理模式

每日一文

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

代理模式


作用

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


代理模式中的角色:

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

两种形式:

  • 静态代理:

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

特点:

代理类和委托类实现了相同的接口,代理类通过委托类实现了相同的方法。这样就出现了大量的代码重复。而且代理类只能为特定的接口(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);
        }
    }

讲解动态代理的原理

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容