设计模式(7) ------------代理模式

概念

代理(Proxy)模式是构造型的设计模式之一,它可以为其他对象提供一种代理(Proxy)以控制对这个对象的访问。

所谓代理,是指具有与代理元(被代理的对象)具有相同的接口的类,客户端必须通过代理与被代理的目标类交互,而代理一般在交互的过程中(交互前后),进行某些特别的处理。

种类

  • Remote Proxy :远程代理。该代理可以让客户端透明地引用一个存在于不同地址空间(远程或本地)的对象。
  • Virtual Proxy :虚拟代理。该代理允许一个对象只有在真正被用到时才被创建。
  • Copy-on-Write Proxy :对象拷贝延迟代理。该代理可以延迟一个对象的拷贝操作到客户调用里,它是Virtual Proxy模式的一种特殊情况。一般来说,对象的深度克隆是一个高开销的动作,该代理可以让这个动作延迟,只有对象被用到的时候才被克隆。
  • Protection Access Proxy :访问保护代理。该代理可以在访问一个对象时附加一些检查操作,比如权限验证等。
  • Cache Proxy :缓存代理。主要为那些创建时高开销的对象提供缓存,以供多客户端共享。
  • Firewall Proxy :防火墙代理。保护目标对象不受某些不良客户端的侵害。
  • Synchronization Proxy :为非同步的目标对象提供并发控制。
  • Smart Reference Proxy :当一个对象被引用时提供某些额外的操作。比如对象被引用时,记录对象被引用的次数等。

类图

Subject 被代理的类的接口。

Proxy 代理类。该代理类实现了Subject接口。

RealSubject 代理元,即被代理的目标类。它实现了Subject接口。

范例1--Protection Access Proxy

下面,我们实现一个Protection Aaccess Proxy:

在该范例中,我们模拟了一个现实应用中使用代理控制对数据库表FILE_TBL的操作。当一个用户具有足够的权限时,则可以进行修改删除等操作。否则,打印权限不够的错误信息。

Client: 测试类

FileTbl: 与数据库表FILE_TBL相对应的数据类

Permission: 权限控制

FileTblDao: 操作数据库表FILE_TBL的DAO接口

FileTblDaoImpl: 操作数据库表FILE_TBL的DAO接口的一个标准实现

FileTblDaoProxy: FileTblDaoImpl的一个代理类。通过该代理类,控制用户进行不同的操作

public class Client {

	public static void main(String[] args) {
		// 只读权限的用户
		Permission searchPermission = new Permission(
				Permission.PERMISSION.SEARCH);
		FileTblDao searchDao = new FileTblDaoProxy(searchPermission);
		FileTbl fileTbl = searchDao.getFile("file01");
		// 只读权限的用户修改文件
		searchDao.updateFile(fileTbl);

		// 全权限的用户
		Permission allPermission = new Permission(Permission.PERMISSION.ALL);
		FileTblDao allDao = new FileTblDaoProxy(allPermission);
		// 全权限的用户修改文件
		allDao.updateFile(fileTbl);
	}

}

/**
 * Subject 操作文件表的DAO。
 */
interface FileTblDao {
	public void deleteFile(FileTbl fileTbl);

	public void updateFile(FileTbl fileTbl);

	public void saveFile(FileTbl fileTbl);

	public FileTbl getFile(String fileId);
}

/**
 * RealSubject 操作文件表的DAO的一个标准实现。
 */
class FileTblDaoImpl implements FileTblDao {

	public void deleteFile(FileTbl fileTbl) {
		System.out.println("delete file:" + fileTbl.getId());
	}

	public void updateFile(FileTbl fileTbl) {
		System.out.println("update file:" + fileTbl.getId());
	}

	public void saveFile(FileTbl fileTbl) {
		System.out.println("save file:" + fileTbl.getId());
	}

	public FileTbl getFile(String fileId) {
		return new FileTbl(fileId);
	}

}

/**
 * Proxy 操作文件表的DAO代理。该DAO根据用户权限判断是否进行修改,删除文件等操作。
 * 如果有足够的权限,则调用FileTblDaoImpl相关方法操作文件。
 */
class FileTblDaoProxy implements FileTblDao {
	FileTblDao fileTblDao;
	Permission permission;

	public FileTblDaoProxy(Permission permission) {
		if (fileTblDao == null) {
			fileTblDao = new FileTblDaoImpl();
		}

		this.permission = permission;
	}

	public void deleteFile(FileTbl fileTbl) {
		if (Permission.PERMISSION.ALL.equals(permission.getLevel())) {
			fileTblDao.deleteFile(fileTbl);
		} else {
			System.out.println("no permission to delete file:"
					+ fileTbl.getId());
		}
	}

	public void updateFile(FileTbl fileTbl) {

		if (Permission.PERMISSION.ALL.equals(permission.getLevel())) {
			fileTblDao.updateFile(fileTbl);
		} else {
			System.out.println("no permission to update file:"
					+ fileTbl.getId());
		}
	}

	public void saveFile(FileTbl fileTbl) {
		if (Permission.PERMISSION.ALL.equals(permission.getLevel())) {
			fileTblDao.saveFile(fileTbl);
		} else {
			System.out.println("no permission to save file:" + fileTbl.getId());
		}
	}

	public FileTbl getFile(String fileId) {
		return fileTblDao.getFile(fileId);
	}
}

/**
 * 文件表类。
 */
class FileTbl {
	private String id;

	public FileTbl(String id) {
		this.id = id;
	}

	public String getId() {
		return id;
	}

	public void setId(String id) {
		this.id = id;
	}

}

/**
 * 权限控制类。
 */
class Permission {
	public static enum PERMISSION {
		ALL, SEARCH
	}

	private PERMISSION level;

	public Permission(PERMISSION level) {
		this.level = level;
	}

	public PERMISSION getLevel() {
		return level;
	}
}

执行Client,输出结果:  
C:\Proxy>javac *.java
C:\Proxy>java Client
no permission to update file:file01
update file:file01

范例2--Cache Proxy

//代理服务器和真实服务器的接口

public interface IServer

{

string Request(string ip,string port,string fileName);

}

//缓存代理服务器

public class CacheProxy : IServer

{

private Hashtable cache = new Hashtable();

//在这个方法里我们实现缓存逻辑

public string Request(string ip, string port,string fileName)

{

string content;

if (cache[string.Format("{0}-{1}-{2}", ip, port, fileName)] == null)

content = new RealServer(ip, port).Request(ip, port, fileName);

cache[string.Format("{0}-{1}-{2}", ip, port, fileName)] = content;

return content;

}

}

//真实的服务器

public class RealServer : IServer

{

public RealServer(string ip, string port)

{

}

public string Request(string ip, string port, string fileName)

{

return string.Format("This is the file's content,file's name is {0}",fileName);

}

}

转载自:http://carver.iteye.com/blog/440141

http://www.cnblogs.com/yuyijq/archive/2008/02/18/1072381.html


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值