当前位置: 首页 > news >正文

精解Java中代理模式的实现

简介摘要: 代理模式是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) { 
           this.username = username; 
           this.password = password; 
   } 

}
  为了提供功能模块,并且希望能够隔离模块,我们需要设计一个接口来定义用户管理模块的接口,这里我们定义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语句 
           System.out.println("Add User Successfully"); 
   } 
   public void removeUser(User user) { 
           //这里处理相关的删除用户的代码任务 
           //比如说连接数据库,执行相关的SQL语句 
           System.out.println("Remove User Successfully"); 
   } 

}
  好了,现在我们对外提供的功能具备了,那么使用者该如何使用这个功能的实现类呢?为了让外界对具体功能类的使用透明化,我们实现一个工厂类来负责创造具体功能模块的对象,并以接口的形式提供外界使用,这样将来更换相关模块的使用将会比较方便。具体工厂类(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(); 
   } 

}

 完成了工厂类的代码,我们可以使用具体模块,这里我们编写一个App.java来使用以下具体功能模块,代码如下:

public class App {

   public static void main(String args[]) { 
           User u = new User(); 
           IUserFace uf = FaceFactory.getInstance().createUserFace(); 
           uf.addUser(u); 
   } 

}
  从上面代码我们可以看到,代码中并没有提及UserFaceImpl这个类,这保证了将来如果需要跟换UserFaceImpl这个类的使用,调用者的代码将不需要做任何的修改。好了,现在我们要来研究一下权限的问题,在这个例子中,我们可能需要在添加用户或者删除用户的时候进行权限检查,符合权限的才能执行相关动作,否则不能执行,那么该如何修改代码才能更加贴切,而且在实际的编写过程中,虽然我们需要权限模块,但有时候为了更好地快速测试,我们常常希望暂时关闭权限模块,如何才能让这样的临时需求变得更加容易处理呢?我们现在使用代理模式来完成这样的任务,现在继续编写一个类叫 UserFaceProxy,让它也实现IUserFace接口,也许你会说,不是已经有一个类实现了这个接口了吗?为什么还要写一个?不要着急,看完这个代码,你就会了解其中的道理了。

public class UserFaceProxy implements IUserFace {

   private IUserFace userFace; 
   public UserFaceProxy(IUserFace userFace) { 
           this.userFace = userFace; 
   } 
   public void addUser(User user) { 
           //在这里检查权限,如果权限不合法则抛出异常
           userFace.addUser(user); 
   } 
   public void removeUser(User user) { 
           //在这里检查权限,如果权限不合法则抛出异常
           //如果权限通过则完成下面的工作 
           userFace.removeUser(user); 
   } 

}
  在代码中你可以看到,这个代理类在构造对象的时候需要传入一个实现了IUserFace接口的类的对象,当代理类对象的方法被调用的时候,首先检查权限,如果权限检查不通过,那么则抛出异常,通过的话则调用构造时传入对象的相应方法]来完成真是的工作。这样的话,我们需要继续修改工厂类的代码如下:

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;就可以了。这就是代理模式在实际中的应用步骤。

本文出自seven的测试人生公众号最新内容请见作者的GitHub页:http://qaseven.github.io/

相关文章:

  • 硅谷投资人Tusk:特朗普当选 硅谷的新机会在哪儿
  • 法国轻奢手机品牌HANMAC 解决手机行业创新瓶颈
  • 易维帮助台如何“重新定义客户服务”?
  • 智慧城市建设新理念
  • 继续走向企业市场 苹果与四大之一德勤合作
  • 依然不可或缺!聊一聊网线的发展历程
  • UOS3.0:给企业以安全感与确定性
  • 《简明电路分析》——1.2节电学主要参数
  • 用户数据是关键 欧盟或调查微软收购领英交易
  • 《PaaS程序设计》一1.6 云计算的承诺(或者炒作)
  • asp.net core 中灵活的配置方式
  • 使用Chrome Headless 快速实现java版数据的抓取
  • 10款中小企业必备的开源免费安全工具
  • 手机线下渠道是最大看点,但不是最重要的着力点
  • 闭源会输但不会死
  • -------------------- 第二讲-------- 第一节------在此给出链表的基本操作
  • @angular/forms 源码解析之双向绑定
  • 【162天】黑马程序员27天视频学习笔记【Day02-上】
  • android高仿小视频、应用锁、3种存储库、QQ小红点动画、仿支付宝图表等源码...
  • C学习-枚举(九)
  • Git学习与使用心得(1)—— 初始化
  • leetcode讲解--894. All Possible Full Binary Trees
  • Traffic-Sign Detection and Classification in the Wild 论文笔记
  • 复杂数据处理
  • 力扣(LeetCode)22
  • 十年未变!安全,谁之责?(下)
  • 物联网链路协议
  • 自制字幕遮挡器
  • ​io --- 处理流的核心工具​
  • $jQuery 重写Alert样式方法
  • (¥1011)-(一千零一拾一元整)输出
  • (1)Map集合 (2)异常机制 (3)File类 (4)I/O流
  • (C++17) optional的使用
  • (echarts)echarts使用时重新加载数据之前的数据存留在图上的问题
  • (zt)最盛行的警世狂言(爆笑)
  • (八)Flask之app.route装饰器函数的参数
  • (动手学习深度学习)第13章 计算机视觉---微调
  • (读书笔记)Javascript高级程序设计---ECMAScript基础
  • (附源码)spring boot公选课在线选课系统 毕业设计 142011
  • (附源码)spring boot基于小程序酒店疫情系统 毕业设计 091931
  • (附源码)ssm基于jsp的在线点餐系统 毕业设计 111016
  • (附源码)计算机毕业设计SSM教师教学质量评价系统
  • (九)One-Wire总线-DS18B20
  • (未解决)jmeter报错之“请在微信客户端打开链接”
  • (转)Spring4.2.5+Hibernate4.3.11+Struts1.3.8集成方案一
  • .NET 反射 Reflect
  • .NET 中选择合适的文件打开模式(CreateNew, Create, Open, OpenOrCreate, Truncate, Append)
  • .net利用SQLBulkCopy进行数据库之间的大批量数据传递
  • .NET中的十进制浮点类型,徐汇区网站设计
  • @RequestParam详解
  • @require_PUTNameError: name ‘require_PUT‘ is not defined 解决方法
  • [.net] 如何在mail的加入正文显示图片
  • [BUUCTF]-PWN:wustctf2020_number_game解析(补码,整数漏洞)
  • [BZOJ]4817: [Sdoi2017]树点涂色
  • [C#]winform利用seetaface6实现C#人脸检测活体检测口罩检测年龄预测性别判断眼睛状态检测