一、引子
我们去科技市场为自己的机器添加点奢侈的配件,很多DIYer 都喜欢去找代理商,因为
在代理商那里拿到的东西不仅质量有保证,而且价格和售后服务上都会好很多。客户通过代
理商得到了自己想要的东西,而且还享受到了代理商额外的服务;而生产厂商通过代理商将
自己的产品推广出去,而且可以将一些销售服务的任务交给代理商来完成(当然代理商要和
厂商来共同分担风险,分配利润),这样自己就可以花更多的心思在产品的设计和生产上了。
在美国,任何企业的产品要想拿到市场上去卖就必须经过代理商这一个环节,否则就是
非法的。看来代理商在商业运作中起着很关键的作用。
不小心把话题扯远了,回过头来,在我们的面向对象的程序设计中,也存在着代理商这
样的角色。下面就跟着这篇文章来看看代理模式的奇妙吧~~~~
二、简介
代理模式有两个英文名字:Proxy Pattern 和 Surrogate Pattern。代理模式的定义为:
为其他对象提供一种代理以控制对这个对象的访问。说白了就是,在一些情况下客户不想或
者不能直接引用一个对象,而代理对象可以在客户和目标对象之间起到中介作用,去掉客户
不能看到的内容和服务或者增添客户需要的额外服务。
根据《Java 与模式》书中对代理模式的分类,代理模式分为8 种,这里将几种常见的、
重要的列举如下:
1) 远程(Remote)代理:为一个位于不同的地址空间的对象提供一个局域代表对象。比
如:你可以将一个在世界某个角落一台机器通过代理假象成你局域网中的一部分。
2) 虚拟(Virtual)代理:根据需要将一个资源消耗很大或者比较复杂的对象延迟的真正需
要时才创建。比如:如果一个很大的图片,需要花费很长时间才能显示出来,那么当这
个图片包含在文档中时,使用编辑器或浏览器打开这个文档,这个大图片可能就影响了
文档的阅读,这时需要做个图片Proxy 来代替真正的图片。
3) 保护(Protect or Access)代理:控制对一个对象的访问权限。比如:在论坛中,不同
的身份登陆,拥有的权限是不同的,使用代理模式可以控制权限(当然,使用别的方式
也可以实现)。
4) 智能引用(Smart Reference)代理:提供比对目标对象额外的服务。比如:纪录访问
的流量(这是个再简单不过的例子),提供一些友情提示等等。
代理模式是一种比较有用的模式,从几个类的“小结构”到庞大系统的“大结构”都可以看
到它的影子。
三、结构
代理模式中的“代理商”要想实现代理任务,就必须和被代理的“厂商”使用共同的接口
(你可以想象为产品)。于是代理模式就有三个角色组成了:
1) 抽象主题角色:声明了真实主题和代理主题的共同接口。
2) 代理主题角色:内部包含对真实主题的引用,并且提供和真实主题角色相同的接口。
3) 真实主题角色:定义真实的对象。
使用类图来表示下三者间的关系如下:
Subject
doSomething()
ProxySubject
RealSubject
doSomething()
doSomething()
anotherThing()
当然,图上所示的是代理模式中的一个具体情况。而代理模式可以非常灵活的使用其他
方式来实现,这样就与图上所示有很大的区别。
也许,现在你已经对代理模式已经有了一个宏观的认识了,下面我们来看看怎么实际的
使用代理模式。
四、举例
以论坛中已注册用户和游客的权限不同来作为第一个例子:已注册的用户拥有发帖,修
改自己的注册信息,修改自己的帖子等功能;而游客只能看到别人发的帖子,没有其他权限。
为了简化代码,更好的显示出代理模式的骨架,我们这里只实现发帖权限的控制。
首先我们先实现一个抽象主题角色 MyForum,里面定义了真实主题和代理主题的共同
接口——发帖功能。
代码如下:
public interface MyForum
{
public void AddFile();
}
这样,真实主题角色和代理主题角色都要实现这个接口。其中真实的主题角色基本就是
将这个接口的方法内容填充进来。所以在这里就不再赘述它的实现。我们把主要的精力放到
关键的代理主题角色上。代理主题角色代码大体如下:
public class MyForumProxy implements MyForum
{
private RealMyForum forum = new RealMyForum() ;
private int permission ; //权限值
public MyForumProxy(int permission)
{
this.permission = permission ;
}
//实现的接口
public void AddFile()
{
//满足权限设置的时候才能够执行操作
//Constants 是一个常量类
if(Constants.ASSOCIATOR == permission)
{
forum.AddFile();
}
else
System.out.println("You are not a associator of MyForum ,please
registe!");
}
}
这样就实现了代理模式的功能。
五、总结
代理模式能够协调调用者和被调用者,能够在一定程度上降低系统的耦合度。代理模式
中的真实主题角色可以结合组合模式来构造,这样一个代理主题角色就可以对一系列的真实
主题角色有效,提高代码利用率,减少不必要子类的产生。
下载:
http://download.csdn.net/detail/undoner/5335717
问题1:代理模式和装饰器模式、适配器模式之间的区别。
答:
尽管Decorator的实现部分与代理相似,但Decorator的目的不一样。Decorator为对象添加一个或多个功能,而代理则控制对对象的访问。
适配器Adapter 为它所适配的对象提供了一个不同的接口。相反,代理提供了与它的实体相同的接口。然而,用于访问保护的代理可能会拒绝执行实体会执行的操作,因此,它的接口实际上可能只是实体接口的一个子集。
问题2:静态代理和动态代理的区别。
答:
静态代理通常只代理一个类,动态代理是代理一个接口下的多个实现类。 静态代理事先知道要代理的是什么,而动态代理不知道要代理什么东西,只有在运行时才知道。 动态代理是实现JDK里的InvocationHandler接口的invoke方法,但注意的是代理的是接口,也就是你的业务类必须要实现接口,通过Proxy里的newProxyInstance得到代理对象。 还有一种动态代理CGLIB,代理的是类,不需要业务类继承接口,通过派生的子类来实现代理。通过在运行时,动态修改字节码达到修改类的目的。