设计模式(三) | 为别人做嫁衣---代理模式
代理模式
代理模式:为其他对象提供一种代理,以控制对这个对象的访问。
模式动机:
-
在某些情况下,一个客户不想或者不能直接引用一个对 象,此时可以通过一个称之为“代理”的第三者来实现 间接引用。代理对象可以在客户端和目标对象之间起到 中介的作用,并且可以通过代理对象去掉客户不能看到 的内容和服务或者添加客户需要的额外服务。
-
代理应用的场合:
-
远程代理:也就是为一个对象在不同的地址空间提供局部代理,这样可以隐藏一个对象存在于不同地址空间的事实。
-
虚拟代理:是根据需要创建很大的对象,通过它来代理来存放实例化需要很长时间的真实对象,例如:网页加载图片。
-
安全代理:用来控制真实对象访问时的权限,一般用于对象应有不同的访问权限的时候。
-
智能指引:是指当调用真实对象时,代理处理另外一些事情。
代码:(通过代理去追妹子的例子):
publicinterfaceIGiveGift{
voidGiveDolls();
voidGiveFlowers();
voidGiveChocolate();
}
//此类中的@Data用到了lombok的注解,可以自动提供get set方法,节省代码量
//请参考“eclipse使用lombok”博客。
importlombok.Data;
public@DataclassSchoolGirl{
Stringname;
}
publicclassPursuitimplementsIGiveGift{
SchoolGirlmm;
publicPursuit(SchoolGirlmm){
super();
this.mm=mm;
}
publicPursuit(){
super();
}
@Override
publicvoidGiveDolls(){
// TODO Auto-generated method stub
System.out.println(mm.getName()+"送你洋娃娃。");
}
@Override
publicvoidGiveFlowers(){
// TODO Auto-generated method stub
System.out.println(mm.getName()+"送你鲜花。");
}
@Override
publicvoidGiveChocolate(){
// TODO Auto-generated method stub
System.out.println(mm.getName()+"送你巧克力。");
}
}
publicclassProxyimplementsIGiveGift{
Pursuitgg;
publicProxy(){
super();
}
publicProxy(SchoolGirlmm){
super();
this.gg=newPursuit(mm);
}
@Override
publicvoidGiveDolls(){
// TODO Auto-generated method stub
gg.GiveDolls();
}
@Override
publicvoidGiveFlowers(){
// TODO Auto-generated method stub
gg.GiveFlowers();
}
@Override
publicvoidGiveChocolate(){
// TODO Auto-generated method stub
gg.GiveChocolate();
}
}
publicclassTest{
publicstaticvoidmain(String[]args){
SchoolGirlmm=newSchoolGirl();
mm.setName("小美");
Proxydaili=newProxy(mm);
daili.GiveDolls();
daili.GiveFlowers();
daili.GiveChocolate();
}
}
代理模式结构图:
- Subject接口,定义了RealSubject和Proxy的共用接口,这样就在任何使用RealSubject的地方都可以使用Proxy。
publicinterfaceSubject
{
publicvoidrequest();
}
-
RealSubject定义Proxy所代表的真实实体,实现了Subject接口。
publicRealSubjectimplementSubject
{
publicvoidrequest()
{
//真实请求
}
}
-
Proxy类,保存一个实体的引用使得代理可以访问实体,并实现Subject的接口,这样就可以用来代替实体。
publicclassProxyimplementSubject
{
RealSubjectrealSubject;
publicProxy(){
realSubject=newRealSubject();
}
publicvoidrequest(){
realSubject.request();
}
}
-
客户端代码:
publicstaticvoidmain(String[]args)
{
Proxyproxy=newProxy();
proxy.request();
}
优点
代理模式的优点
-
代理模式能够协调调用者和被调用者,在一定程度上降低了系 统的耦合度。
-
远程代理使得客户端可以访问在远程机器上的对象,远程机器 可能具有更好的计算性能与处理速度,可以快速响应并处理客户端请求。
-
虚拟代理通过使用一个小对象来代表一个大对象,可以减少系 统资源的消耗,对系统进行优化并提高运行速度。
-
保护代理可以控制对真实对象的使用权限。
缺点
代理模式的缺点
-
由于在客户端和真实主题之间增加了代理对象,因此 有些类型的代理模式可能会造成请求的处理速度变慢。
-
实现代理模式需要额外的工作,有些代理模式的实现 非常复杂。
适用环境
根据代理模式的使用目的,常见的代理模式有以下几种类型:
-
远程(Remote)代理:为一个位于不同的地址空间的对象提供一个本地 的代理对象,这个不同的地址空间可以是在同一台主机中,也可是在 另一台主机中,远程代理又叫做大使(Ambassador)。
-
虚拟(Virtual)代理:如果需要创建一个资源消耗较大的对象,先创建一个消耗相对较小的对象来表示,真实对象只在需要时才会被真正创建。
-
Copy-on-Write代理:它是虚拟代理的一种,把复制(克隆)操作延迟 到只有在客户端真正需要时才执行。一般来说,对象的深克隆是一个 开销较大的操作,Copy-on-Write代理可以让这个操作延迟,只有对象被用到的时候才被克隆。
-
保护(Protect or Access)代理:控制对一个对象的访问,可以给不同的用户提供不同级别的使用权限。
-
缓冲(Cache)代理:为某一个目标操作的结果提供临时的存储空间,以便多个客户端可以共享这些结果。
-
防火墙(Firewall)代理:保护目标不让恶意用户接近。
-
同步化(Synchronization)代理:使几个用户能够同时使用一个对象而没有冲突。
-
智能引用(Smart Reference)代理:当一个对象被引用时,提供一些额外的操作,如将此对象被调用的次数记录下来等。