简介摘要 代理模式是GOF设计模式中的一种常用于权限模块的架构设计其根本的原理是通过将一个代理对象交给调用者使得调用者不能直接使用相应的功能模块所 有的调用被传递给代理对象代理对象负责对真实模块完成调用在调用者与被调用者之间建立了一个隔离带我们可以使 代理模式是GOF设计模式中的一种常用于权限模块的架构设计其根本的原理是通过将一个代理对象交给调用者使得调用者不能直接使用相应的功能模块所有的调用被传递给代理对象代理对象负责对真实模块完成调用在调用者与被调用者[bei tiao yong zhe]之间建立了一个隔离带我们可以使用这个隔离带进行权限检查对象的延迟[yan chi] 加载等功能的实现这里不对这个设计模式的具体原理多加解释[jie shi]我们直接通过一个实例的编写来完成对代理模式的应用[ying yong]在理解了代理模式之后我们将继续介绍 java中提供的一种动态[dong tai]代理技术与其实现
这里我们假设有一个用户管理模块这个模块提供了添加用户删除用户的功能我们现在要使用代理模式来检查权限该如何实现呢?首先我们需要具有一个类叫User用来表示一个用户的信息[xin xi]代码如下
public class User {
private String username;
private String password;
public User() {
}
public User(String username String password) {
thisusername = username;
thispassword = password;
}
}
为了提供功能模块并且希望[xi wang]能够隔离模块我们需要设计一个接口来定义用户管理模块的接口这里我们定义IUserFace接口代码如下
public interface IUserFace {
public void addUser(User user);
public void removeUser(User user);
}
接下来为这个接口编写一个真正实现具体功能的类出来定义为UserFaceImpl代码如下
public class UserFaceImpl implements IUserFace {
public void addUser(User user) {
//这里处理相关的添加用户的代码任务
//比如说连接数据库执行相关的SQL语句
Systemoutprintln(Add User Successfully);
}
public void removeUser(User user) {
//这里处理相关的删除用户的代码任务
//比如说连接数据库执行相关的SQL语句
Systemoutprintln(Remove User Successfully);
}
}
好了现在我们对外提供的功能具备了那么使用者该如何使用这个功能的实现类呢?为了让外界对具体功能类的使用透明[tou ming]化我们实现一个工厂类来负责创造具体功能模块的对象并以接口的形式提供外界使用这样将来更换相关模块的使用将会比较方便具体工厂类(FaceFactory)代码如下
public class FaceFactory {
private static FaceFactory instance;
private FaceFactory() {
}
public static FaceFactory getInstance() {
if(instance == null) {
instance = new FaceFactory();
}
return instance;
}
public IUserFace createUserFace() {
return new UserFaceImpl();
}
}
完成了工厂类的代码我们可以使用具体模块这里我们编写一个Appjava来使用以下具体功能模块代码如下
public class App {
public static void main(String args[]) {
User u = new User();
IUserFace uf = FaceFactorygetInstance()createUserFace();
ufaddUser(u);
}
}
从上面代码我们可以看到代码中并没有提及UserFaceImpl这个类这保证了将来如果需要跟换UserFaceImpl这个类的使用调用者的代码将不需要做任何的修改[xiu gai]好了现在我们要来研究一下权限的问题[wen ti]在这个例子中我们可能需要在添加用户或者删除用户的时候进行权限检查符合权限的才能执行相关动作否则不能执行那么该如何修改[xiu gai]代码才能更加贴切而且在实际的编写过程中虽然我们需要权限模块但有时候为了更好地快速测试我们常常希望暂时关闭权限模块如何才能让这样的临时需求[xu qiu]变得更加容易处理呢?我们现在使用代理模式来完成这样的任务现在继续编写一个类叫 UserFaceProxy让它也实现IUserFace接口也许你会说不是已经有一个类实现了这个接口了吗?为什么还要写一个?不要着急看完这个代码你就会了解其中的道理了
public class UserFaceProxy implements IUserFace {
private IUserFace userFace;
public UserFaceProxy(IUserFace userFace) {
thisuserFace = userFace;
}
public void addUser(User user) {
//在这里检查权限如果权限不合法则抛出[pao chu]异常[yi chang][pao chu yi chang]
//如果权限通过则完成下面的工作
userFaceaddUser(user);
}
public void removeUser(User user) {
//在这里检查权限如果权限不合法则抛出[pao chu]异常[yi chang][pao chu yi chang]
//如果权限通过则完成下面的工作
userFaceremoveUser(user);
}
}
在代码中你可以看到这个代理类在构造对象的时候需要传入一个实现了IUserFace接口的类的对象当代理类对象的方法[fang fa]被调用的时候首先检查权限如果权限检查不通过那么则抛出[pao chu]异常[yi chang][pao chu yi chang]通过的话则调用构造时传入对象的相应方法[fang fa]来完成真是的工作这样的话我们需要继续修改[xiu gai]工厂类的代码如下
public class FaceFactory {
private static FaceFactory instance;
private FaceFactory() {
}
public static FaceFactory getInstance() {
if(instance == null) {
instance = new FaceFactory();
}
return instance;
}
public IUserFace createUserFace() {
IUserFace userFace = new UserFaceImpl();
IUserFace proxy = new UserFaceProxy(userFace);
return proxy;
}
}
好了到这里你是不是已经明白了?通过这样的代理模式我们完成了权限检查的隔离处理当需要临时关闭权限检查的时候我们只需要在如上的代码中return userFace;就可以了这就是代理模式在实际中的应用[ying yong]步骤