简介

  • 代理模式有两个英文名字:Proxy Pattern 和 Surrogate Pattern。

  • 代理模式的定义为: 为其他对象提供一种代理以控制对这个对象的访问。

  • 说白了就是,在一些情况下客户不想或 者不能直接引用一个对象,而代理对象可以在客户和目标对象之间起到中介作用,去掉客户 不能看到的内容和服务或者增添客户需要的额外服务。

  • 根据《Java 与模式》书中对代理模式的分类,代理模式分为 8 种,这里将几种常见的、 重要的列举如下:

    1. 远程(Remote)代理:为一个位于不同的地址空间的对象提供一个局域代表对象。
      • 比 如:你可以将一个在世界某个角落一台机器通过代理假象成你局域网中的一部分。
    2. 虚拟(Virtual)代理:根据需要将一个资源消耗很大或者比较复杂的对象延迟的真正需 要时才创建。
      • 比如:如果一个很大的图片,需要花费很长时间才能显示出来,那么当这 个图片包含在文档中时,使用编辑器或浏览器打开这个文档,这个大图片可能就影响了 文档的阅读,这时需要做个图片 Proxy 来代替真正的图片。
    3. 保护(Protect or Access)代理:控制对一个对象的访问权限。
      • 比如:在论坛中,不同 的身份登陆,拥有的权限是不同的,使用代理模式可以控制权限(当然,使用别的方式 也可以实现)。
    4. 智能引用(Smart Reference)代理:提供比对目标对象额外的服务。
      • 比如:纪录访问 的流量(这是个再简单不过的例子),提供一些友情提示等等。 代理模式是一种比较有用的模式,从几个类的“小结构”到庞大系统的“大结构”都可以看 到它的影子。
  • 什么是代理模式呢?我很忙,忙的没空理你,那你要找我呢就先找我的代理人吧,那代理人总要知道被代理人能做哪些事情不能做哪些事情吧,那就是两个人具备同一个接口,代理人虽然不能干活,但是被代理的人能干活呀。

架构图

举例

比如西门庆找潘金莲,那潘金莲不好意思答复呀,咋办,找那个王婆做代理,表现在程序上时这样的:先定义一种类型的女人:

1
2
3
4
5
6
7
8
9
package com.cbf4life.proxy;
/**
* 定义一种类型的女人,王婆和潘金莲都属于这个类型的女人
*/
public interface KindWomen {
//这种类型的女人能做什么事情呢?
public void makeEyesWithMan(); //抛媚眼
public void happyWithMan(); //happy what? You know that!
}

一种类型嘛,那肯定是接口,然后定义潘金莲:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
package com.cbf4life.proxy;

/**
* I'm glad to share my knowledge with you all.
* 定一个潘金莲是什么样的人
*/

public class PanJinLian implements KindWomen {

public void happyWithMan() {
System.out.println("潘金莲在和男人做那个.....");
}

public void makeEyesWithMan() {
System.out.println("潘金莲抛媚眼");
}
}

再定一个丑陋的王婆:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package com.cbf4life.proxy;
/**
* 王婆这个人老聪明了,她太老了,是个男人都看不上,
* 但是她有智慧有经验呀,她作为一类女人的代理!
*/
public class WangPo implements KindWomen {
private KindWomen kindWomen;

public WangPo(){ //默认的话,是潘金莲的代理
this.kindWomen = new PanJinLian();
}

//她可以是KindWomen的任何一个女人的代理,只要你是这一类型
public WangPo(KindWomen kindWomen){
this.kindWomen = kindWomen;
}

public void happyWithMan() {
this.kindWomen.happyWithMan(); //自己老了,干不了,可以让年轻的代替
}

public void makeEyesWithMan() {
this.kindWomen.makeEyesWithMan(); //王婆这么大年龄了,谁看她抛媚眼?!
}
}

两个女主角都上场了,男主角也该出现了:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
package com.cbf4life.proxy;
/**
* 定义一个西门庆,这人色中饿鬼
*/
public class XiMenQing {
/*
* 水浒里是这样写的:西门庆被潘金莲用竹竿敲了一下难道,痴迷了,
* 被王婆看到了, 就开始撮合两人好事,王婆作为潘金莲的代理人
* 收了不少好处费,那我们假设一下:
* 如果没有王婆在中间牵线,这两个不要脸的能成吗?难说的很!
*/
public static void main(String[] args) {
//把王婆叫出来
WangPo wangPo = new WangPo();
//然后西门庆就说,我要和潘金莲happy,然后王婆就安排了西门庆丢筷子的那出戏:
wangPo.makeEyesWithMan(); //看到没,虽然表面上时王婆在做,实际上是潘金莲
wangPo.happyWithMan(); }
}

那这就是活生生的一个例子,通过代理人实现了某种目的,如果真去掉王婆这个中间环节,直接是西门庆和潘金莲勾搭,估计很难成就武松杀嫂事件。

那我们再考虑一下,水浒里还有没有这类型的女人?有,卢俊义的老婆贾氏(就是和那个固管家苟合的那个),这名字起的:“假使”,那我们也让王婆做她的代理:

把贾氏素描出来:

1
2
3
4
5
6
7
8
9
10
11
12
package com.cbf4life.proxy;
/**
* I'm glad to share my knowledge with you all.
*/
public class JiaShi implements KindWomen {
public void happyWithMan() {
System.out.println("贾氏正在Happy中......");
}
public void makeEyesWithMan() {
System.out.println("贾氏抛媚眼");
}
}

西门庆勾贾氏:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
西门庆勾贾氏:
package com.cbf4life.proxy;
/**
* I'm glad to share my knowledge with you all.
* 定义一个西门庆,这人色中饿鬼
*/
public class XiMenQing {
public static void main(String[] args) {
//改编一下历史,贾氏被西门庆勾走:
JiaShi jiaShi = new JiaShi();
WangPo wangPo = new WangPo(jiaShi); //让王婆作为贾氏的代理人
wangPo.makeEyesWithMan();
wangPo.happyWithMan();
}
}

说完这个故事,那额总结一下,代理模式主要使用了Java 的多态,干活的是被代理类,代理类主要是接活,你让我干活,好,我交给幕后的类去干,你满意就成,那怎么知道被代理类能不能干呢?同根就成,

大家知根知底,你能做啥,我能做啥都清楚的很,同一个接口呗。