代理模式结构
- 抽象主题(Subject)角色 声明了真实主题和代理主题的共同接口,这样一来在任何使用真实主题的地方都可以使用代理主题。
- 代理主题(Proxy)角色 代理主题角色内部含有对真实主题的引用,从而可以再任何时候操作真实主题对象;==代理主题角色提供一个与真实主题角色相同的接口,以便可以再任何时候都可以替代真实主题;==控制真实主题的应用,负责在需要的时候创建真实主题对象(和删除主题对象)
- 真实主题(RealSubject)角色 定义了代理角色所代表的真实对象
代理模式
- 代理模式(proxy)模式给某一个对象提供一个替身或占位符,以控制对这个对象的访问。
- 所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动。在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介作用。
代理模式应用
一、远程(Remote)代理
为一个位于不同的地址空间的对象提供一个局域代表对象。可以隐藏一个对象存在于不同地址空间的事实。这个不同的地址空间可以使在本机器中,也可是在另一台机器中。远程代理又叫做大使。
可以将网络的细节隐藏起来,使得客户端不必考虑网络的存在。客户完全可以认为被代理的对象是局域的而不是远程的,而代理对象承担了大部分的网络通信工作。
二、虚拟(Virtual)代理
代理对象可以在必要的时候才将被代理的对象加载。代理可以对加载过程加以必要的优化。当一个模块的加载十分耗费资源的时候,虚拟代理的优点就非常明显。
应用:
- 某个客户端不能直接操作到某个对象,担又必须和那个对象有所互动。
- 对于开销很大的对象,只有在使用它是才创建,这个原则可以为为我们节省很多宝贵的内存资源。
三、保护(Protect or Access)代理(安全代理)
可以在运行时对用户的有关权限进行检查,然后在核实后决定将调用传递给被代理的对象。
授权机制:不同级别的用户对同一对象拥有不同的访问权利。
四、智能引用(Smart Reference)代理
当一个对象呗引用是,提供一些额外的曹锁,比如讲对此对象调用的次数记录下来等。
五、不常用代理
- Copy-on-Wright代理
虚拟代理的一种。把复制(克隆)拖延到只有在哭护短需要时,才真正采取行动。 - Cache代理
为那些昂贵操作的结果提供临时的存储空间,一遍多个客户端可以共享这些结果。 - 防火墙(Friewall)代理
保护目标,不让恶意用户接近。 - 同步化(Synchronization)代理
使几个用户能够同时访问目标对象而没有冲突。
举个例子
【问题】
订单处理系统。一单订单被创建,只有订单的创建人才可以修改订单中的数据,其他人则不能修改。一共有三个属性:产品名称、订购数量、订购人。
【代码】
using System;
namespace Order
{
//定义接口
public interface OrderApi
{
public String getProductName();
public void setProductName(String productName, String user);
public int getOrderNum();
public void setOrderNum(int orderNum, String user);
public String getOrderName();
public void setOrderName(String orderName, String user);
}
//订单对象定义
public class Order : OrderApi
{
private String productName;
private int orderNum;
private String orderName;
public Order(String productName, int orderNum, String orderName)
{
this.productName = productName;
this.orderNum = orderNum;
this.orderName = orderName;
}
public String getOrderName()
{
return orderName;
}
public int getOrderNum()
{
return orderNum;
}
public String getProductName()
{
return productName;
}
public void setOrderName(String orderName, String user)
{
this.orderName = orderName;
}
public void setOrderNum(int orderNum, String user)
{
this.orderNum = orderNum;
}
public void setProductName(String productName, String user)
{
this.productName = productName;
}
}
//代理端
public class OrdeProxy : OrderApi
{
private Order order = null;
public OrdeProxy(Order order)
{
this.order = order;
}
public string getOrderName()
{
return this.order.getOrderName();
}
public int getOrderNum()
{
return this.order.getOrderNum();
}
public string getProductName()
{
return this.order.getProductName();
}
public void setOrderName(string orderName, string user)
{
if (user != null && user.Equals(this.getOrderName()))
this.order.setOrderName(orderName, user);
else
Console.WriteLine("对不起" + user + ",您无权修改订单中的订购人!");
}
public void setOrderNum(int orderNum, string user)
{
if (user != null && user.Equals(this.getOrderName()))
this.order.setOrderNum(orderNum, user);
else
Console.WriteLine("对不起" + user + ",您无权修改订单中的订单数量!");
}
public void setProductName(string productName, string user)
{
if (user != null && user.Equals(this.getOrderName()))
this.order.setProductName(productName, user);
else
Console.WriteLine("对不起" + user + ",您无权修改订单中的产品名称!");
}
public String toString()
{
return "ProductName=" + this.getProductName() + ",OrderName=" + this.getOrderName() + ",OrderNum=" + this.getOrderNum();
}
}
//客户端
class Program
{
static void Main(string[] args)
{
Order od = new Order("设计模式", 100, "张三");
OrdeProxy order = new OrdeProxy(od);
order.setOrderNum(123, "李四");
Console.WriteLine("李四修改订单后没有变化" + order.toString());
order.setOrderNum(123, "张三");
Console.WriteLine("张三修改订单后,订单记录:" + order.toString());
}
}
}
【UML图】