【转】 代理模式

原文链接:http://layznet.iteye.com/blog/1182924
一、代理概念
为某个对象提供一个代理,以控制对这个对象的访问。 代理类和委托类有共同的父类或父接口,这样在任何使用委托类对象的地方都可以用代理对象替代。代理类负责请求的预处理、过滤、将请求分派给委托类处理、以及委托类执行完请求后的后续处理。

图1:代理模式

从图中可以看出,代理接口(Subject)、代理类(ProxySubject)、委托类(RealSubject)形成一个“品”字结构。
根据代理类的生成时间不同可以将代理分为静态代理和动态代理两种。

下面以一个模拟需求说明静态代理和动态代理:委托类要处理一项耗时较长的任务,客户类需要打印出执行任务消耗的时间。解决这个问题需要记录任务执行前时间和任务执行后时间,两个时间差就是任务执行消耗的时间。

二、静态代理
由程序员创建或工具生成代理类的源码,再编译代理类。所谓静态也就是在程序运行前就已经存在代理类的字节码文件,代理类和委托类的关系在运行前就确定了。
清单1:代理接口

Java代码  收藏代码
  1. /**  
  2.  * 代理接口。处理给定名字的任务。 
  3.  */  
  4. public interface Subject {  
  5.   /** 
  6.    * 执行给定名字的任务。 
  7.     * @param taskName 任务名 
  8.    */  
  9.    public void dealTask(String taskName);   
  10. }  


清单2:委托类,具体处理业务。

Java代码  收藏代码
  1. /** 
  2.  * 真正执行任务的类,实现了代理接口。 
  3.  */  
  4. public class RealSubject implements Subject {  
  5.   
  6.  /** 
  7.   * 执行给定名字的任务。这里打印出任务名,并休眠500ms模拟任务执行了很长时间 
  8.   * @param taskName  
  9.   */  
  10.    @Override  
  11.    public void dealTask(String taskName) {  
  12.       System.out.println("正在执行任务:"+taskName);  
  13.       try {  
  14.          Thread.sleep(500);  
  15.       } catch (InterruptedException e) {  
  16.          e.printStackTrace();  
  17.       }  
  18.    }  
  19. }  


清单3:静态代理类

Java代码  收藏代码
  1. /** 
  2.  * 代理类,实现了代理接口。 
  3.  */  
  4. public class ProxySubject implements Subject {  
  5.  //代理类持有一个委托类的对象引用  
  6.  private Subject delegate;  
  7.    
  8.  public ProxySubject(Subject delegate) {  
  9.   this.delegate = delegate;  
  10.  }  
  11.   
  12.  /** 
  13.   * 将请求分派给委托类执行,记录任务执行前后的时间,时间差即为任务的处理时间 
  14.   *  
  15.   * @param taskName 
  16.   */  
  17.  @Override  
  18.  public void dealTask(String taskName) {  
  19.   long stime = System.currentTimeMillis();   
  20.   //将请求分派给委托类处理  
  21.   delegate.dealTask(taskName);  
  22.   long ftime = System.currentTimeMillis();   
  23.   System.out.println("执行任务耗时"+(ftime - stime)+"毫秒");  
  24.     
  25.  }  
  26. }  


清单4:生成静态代理类工厂

Java代码  收藏代码
  1. public class SubjectStaticFactory {  
  2.  //客户类调用此工厂方法获得代理对象。  
  3.  //对客户类来说,其并不知道返回的是代理类对象还是委托类对象。  
  4.  public static Subject getInstance(){   
  5.   return new ProxySubject(new RealSubject());  
  6.  }  
  7. }  


清单5:客户类

Java代码  收藏代码
  1. public class Client1 {  
  2.   
  3.  public static void main(String[] args) {  
  4.   Subject proxy = SubjectStaticFactory.getInstance();  
  5.   proxy.dealTask("DBQueryTask");  
  6.  }   
  7.   
  8. }  



静态代理类优缺点
优点:业务类只需要关注业务逻辑本身,保证了业务类的重用性。这是代理的共有优点。
缺点:
1)代理对象的一个接口只服务于一种类型的对象,如果要代理的方法很多,势必要为每一种方法都进行代理,静态代理在程序规模稍大时就无法胜任了。
2)如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也需要实现此方法。增加了代码维护的复杂度。

三、动态代理
动态代理类的源码是在程序运行期间由JVM根据反射等机制动态的生成,所以不存在代理类的字节码文件。代理类和委托类的关系是在程序运行时确定。

1、先看看与动态代理紧密关联的Java API。
1)java.lang.reflect.Proxy
这是 Java 动态代理机制生成的所有动态代理类的父类,它提供了一组静态方法来为一组接口动态地生成代理类及其对象。
清单6:Proxy类的静态方法

Java代码  收藏代码
  1. // 方法 1: 该方法用于获取指定代理对象所关联的调用处理器  
  2. static InvocationHandler getInvocationHandler(Object proxy)   
  3.   
  4. // 方法 2:该方法用于获取关联于指定类装载器和一组接口的动态代理类的类对象  
  5. static Class getProxyClass(ClassLoader loader, Class[] interfaces)   
  6.   
  7. // 方法 3:该方法用于判断指定类对象是否是一个动态代理类  
  8. static boolean isProxyClass(Class cl)   
  9.   
  10. // 方法 4:该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例  
  11. static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)   



2)java.lang.reflect.InvocationHandler
这是调用处理器接口,它自定义了一个 invoke 方法,用于集中处理在动态代理类对象上的方法调用,通常在该方法中实现对委托类的代理访问。每次生成动态代理类对象时都要指定一个对应的调用处理器对象。
清单7:InvocationHandler的核心方法

Java代码  收藏代码
  1. // 该方法负责集中处理动态代理类上的所有方法调用。第一个参数既是代理类实例,第二个参数是被调用的方法对象  
  2. // 第三个方法是调用参数。调用处理器根据这三个参数进行预处理或分派到委托类实例上反射执行  
  3. Object invoke(Object proxy, Method method, Object[] args)   



3)java.lang.ClassLoader
这是类装载器类,负责将类的字节码装载到 Java 虚拟机(JVM)中并为其定义类对象,然后该类才能被使用。Proxy 静态方法生成动态代理类同样需要通过类装载器来进行装载才能使用,它与普通类的唯一区别就是其字节码是由 JVM 在运行时动态生成的而非预存在于任何一个 .class 文件中。
每次生成动态代理类对象时都需要指定一个类装载器对象

2、动态代理实现步骤
具体步骤是:
a. 实现InvocationHandler接口创建自己的调用处理器
b. 给Proxy类提供ClassLoader和代理接口类型数组创建动态代理类
c. 以调用处理器类型为参数,利用反射机制得到动态代理类的构造函数
d. 以调用处理器对象为参数,利用动态代理类的构造函数创建动态代理类对象
清单8:分步骤实现动态代理

Java代码  收藏代码
  1. // InvocationHandlerImpl 实现了 InvocationHandler 接口,并能实现方法调用从代理类到委托类的分派转发  
  2. // 其内部通常包含指向委托类实例的引用,用于真正执行分派转发过来的方法调用  
  3. InvocationHandler handler = new InvocationHandlerImpl(..);   
  4.   
  5. // 通过 Proxy 为包括 Interface 接口在内的一组接口动态创建代理类的类对象  
  6. Class clazz = Proxy.getProxyClass(classLoader, new Class[] { Interface.class, ... });   
  7.   
  8. // 通过反射从生成的类对象获得构造函数对象  
  9. Constructor constructor = clazz.getConstructor(new Class[] { InvocationHandler.class });   
  10.   
  11. // 通过构造函数对象创建动态代理类实例  
  12. Interface Proxy = (Interface)constructor.newInstance(new Object[] { handler });   



Proxy类的静态方法newProxyInstance对上面具体步骤的后三步做了封装,简化了动态代理对象的获取过程。
清单9:简化后的动态代理实现

Java代码  收藏代码
  1. // InvocationHandlerImpl 实现了 InvocationHandler 接口,并能实现方法调用从代理类到委托类的分派转发  
  2. InvocationHandler handler = new InvocationHandlerImpl(..);   
  3.   
  4. // 通过 Proxy 直接创建动态代理类实例  
  5. Interface proxy = (Interface)Proxy.newProxyInstance( classLoader,   
  6.      new Class[] { Interface.class },  handler );   



3、动态代理实现示例
清单10:创建自己的调用处理器

Java代码  收藏代码
  1. /** 
  2.  * 动态代理类对应的调用处理程序类 
  3.  */  
  4. public class SubjectInvocationHandler implements InvocationHandler {  
  5.    
  6.  //代理类持有一个委托类的对象引用  
  7.  private Object delegate;  
  8.    
  9.  public SubjectInvocationHandler(Object delegate) {  
  10.   this.delegate = delegate;  
  11.  }  
  12.    
  13.  @Override  
  14.  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
  15.   long stime = System.currentTimeMillis();   
  16.   //利用反射机制将请求分派给委托类处理。Method的invoke返回Object对象作为方法执行结果。  
  17.   //因为示例程序没有返回值,所以这里忽略了返回值处理  
  18.   method.invoke(delegate, args);  
  19.   long ftime = System.currentTimeMillis();   
  20.   System.out.println("执行任务耗时"+(ftime - stime)+"毫秒");  
  21.     
  22.   return null;  
  23.  }  
  24. }   


清单11:生成动态代理对象的工厂,工厂方法列出了如何生成动态代理类对象的步骤。

Java代码  收藏代码
  1. /** 
  2.  * 生成动态代理对象的工厂. 
  3.  */  
  4. public class DynProxyFactory {  
  5.  //客户类调用此工厂方法获得代理对象。  
  6.  //对客户类来说,其并不知道返回的是代理类对象还是委托类对象。  
  7.  public static Subject getInstance(){   
  8.   Subject delegate = new RealSubject();  
  9.   InvocationHandler handler = new SubjectInvocationHandler(delegate);  
  10.   Subject proxy = null;  
  11.   proxy = (Subject)Proxy.newProxyInstance(  
  12.     delegate.getClass().getClassLoader(),   
  13.     delegate.getClass().getInterfaces(),   
  14.     handler);  
  15.   return proxy;  
  16.  }  
  17. }  



清单12:动态代理客户类

Java代码  收藏代码
  1. public class Client {  
  2.   
  3.  public static void main(String[] args) {  
  4.   
  5.   Subject proxy = DynProxyFactory.getInstance();  
  6.   proxy.dealTask("DBQueryTask");  
  7.  }   
  8.   
  9. }  



4、动态代理机制特点 
首先是动态生成的代理类本身的一些特点。1)包:如果所代理的接口都是 public 的,那么它将被定义在顶层包(即包路径为空),如果所代理的接口中有非 public 的接口(因为接口不能被定义为 protect 或 private,所以除 public 之外就是默认的 package 访问级别),那么它将被定义在该接口所在包(假设代理了 com.ibm.developerworks 包中的某非 public 接口 A,那么新生成的代理类所在的包就是 com.ibm.developerworks),这样设计的目的是为了最大程度的保证动态代理类不会因为包管理的问题而无法被成功定义并访问;2)类修 饰符:该代理类具有 final 和 public 修饰符,意味着它可以被所有的类访问,但是不能被再度继承;3)类名:格式是“$ProxyN”,其中 N 是一个逐一递增的阿拉伯数字,代表 Proxy 类第 N 次生成的动态代理类,值得注意的一点是,并不是每次调用 Proxy 的静态方法创建动态代理类都会使得 N 值增加,原因是如果对同一组接口(包括接口排列的顺序相同)试图重复创建动态代理类,它会很聪明地返回先前已经创建好的代理类的类对象,而不会再尝试去创 建一个全新的代理类,这样可以节省不必要的代码重复生成,提高了代理类的创建效率。4)类继承关系:该类的继承关系如图:

图2:动态代理类的继承关系



由图可见,Proxy 类是它的父类,这个规则适用于所有由 Proxy 创建的动态代理类。而且该类还实现了其所代理的一组接口,这就是为什么它能够被安全地类型转换到其所代理的某接口的根本原因。

接下来让我们了解一下代理类实例的一些特点。每个实例都会关联一个调用处理器对象,可以通过 Proxy 提供的静态方法 getInvocationHandler 去获得代理类实例的调用处理器对象。在代理类实例上调用其代理的接口中所声明的方法时,这些方法最终都会由调用处理器的 invoke 方法执行,此外,值得注意的是,代理类的根类 java.lang.Object 中有三个方法也同样会被分派到调用处理器的 invoke 方法执行,它们是 hashCode,equals 和 toString,可能的原因有:一是因为这些方法为 public 且非 final 类型,能够被代理类覆盖;二是因为这些方法往往呈现出一个类的某种特征属性,具有一定的区分度,所以为了保证代理类与委托类对外的一致性,这三个方法也应 该被分派到委托类执行。当代理的一组接口有重复声明的方法且该方法被调用时,代理类总是从排在最前面的接口中获取方法对象并分派给调用处理器,而无论代理 类实例是否正在以该接口(或继承于该接口的某子接口)的形式被外部引用,因为在代理类内部无法区分其当前的被引用类型。

接着来了解一下被代理的一组接口有哪些特点。首先,要注意不能有重复的接口,以避免动态代理类代码生成时的编译错误。其次,这些接口对于类装载器 必须可见,否则类装载器将无法链接它们,将会导致类定义失败。再次,需被代理的所有非 public 的接口必须在同一个包中,否则代理类生成也会失败。最后,接口的数目不能超过 65535,这是 JVM 设定的限制。

最后再来了解一下异常处理方面的特点。从调用处理器接口声明的方法中可以看到理论上它能够抛出任何类型的异常,因为所有的异常都继承于 Throwable 接口,但事实是否如此呢?答案是否定的,原因是我们必须遵守一个继承原则:即子类覆盖父类或实现父接口的方法时,抛出的异常必须在原方法支持的异常列表之 内。所以虽然调用处理器理论上讲能够,但实际上往往受限制,除非父接口中的方法支持抛 Throwable 异常。那么如果在 invoke 方法中的确产生了接口方法声明中不支持的异常,那将如何呢?放心,Java 动态代理类已经为我们设计好了解决方法:它将会抛出 UndeclaredThrowableException 异常。这个异常是一个 RuntimeException 类型,所以不会引起编译错误。通过该异常的 getCause 方法,还可以获得原来那个不受支持的异常对象,以便于错误诊断。

5、动态代理的优点和美中不足
优点:
动态代理与静态代理相比较,最大的好处是接口中声明的所有方法都被转移到调用处理器一个集中的方法中处理 (InvocationHandler.invoke)。这样,在接口方法数量比较多的时候,我们可以进行灵活处理,而不需要像静态代理那样每一个方法进 行中转。在本示例中看不出来,因为invoke方法体内嵌入了具体的外围业务(记录任务处理前后时间并计算时间差),实际中可以类似Spring AOP那样配置外围业务。

美中不足:
诚然,Proxy 已经设计得非常优美,但是还是有一点点小小的遗憾之处,那就是它始终无法摆脱仅支持 interface 代理的桎梏,因为它的设计注定了这个遗憾。回想一下那些动态生成的代理类的继承关系图,它们已经注定有一个共同的父类叫 Proxy。Java 的继承机制注定了这些动态代理类们无法实现对 class 的动态代理,原因是多继承在 Java 中本质上就行不通。

有很多条理由,人们可以否定对 class 代理的必要性,但是同样有一些理由,相信支持 class 动态代理会更美好。接口和类的划分,本就不是很明显,只是到了 Java 中才变得如此的细化。如果只从方法的声明及是否被定义来考量,有一种两者的混合体,它的名字叫抽象类。实现对抽象类的动态代理,相信也有其内在的价值。此 外,还有一些历史遗留的类,它们将因为没有实现任何接口而从此与动态代理永世无缘。如此种种,不得不说是一个小小的遗憾。

转载于:https://www.cnblogs.com/bingzhikun/p/4797679.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/419578.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

oracle之单行函数之多表查询值之课后练习

26. 多表连接查询时, 若两个表有同名的列, 必须使用表的别名对列名进行引用, 否则出错!27. 查询出公司员工的 last_name, department_name, cityselect last_name, department_name, cityfrom departments d, employees e, locations lwhere d.department_id e.department_id …

行末没有空格c语言,新人提问:如何将输出时每行最后一个空格删除

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼如何将每行最后一个空格删除&#xff0c;使矩阵只有数字间有空格&#xff0c;没有多余空格&#xff1f;#include#includeint main(){int i,j,k,m,n,x,h,y;int a[15][15]{0};while(scanf("%d",&i)){k1;for(n1;n<i;…

Core Animation

关键字 1.Core Animation的核心类是CALayer,通过对其属性进行配置可以展现不同的外观&#xff0c;这些属性包括位置&#xff0c;尺寸&#xff0c;图片内容&#xff0c;背景色&#xff0c;边界&#xff0c;阴影&#xff0c;以及角半径。 CATextLayer *textLayer;textLayer [CAT…

oracle之单行函数之分组函数

--分组函数 select avg(salary),max(salary),min(salary),sum(salary) from employees 运行结果 --判断大小 select max(last_name),min(last_name),max(hire_date),min(hire_date) from employees 运行结果 --计数 select count(employee_id),count(last_name),count(hire_da…

LeetCode 98. Validate Binary Search Tree

原题链接在这里&#xff1a;https://leetcode.com/problems/validate-binary-search-tree/ 题目&#xff1a; Given a binary tree, determine if it is a valid binary search tree (BST). Assume a BST is defined as follows: The left subtree of a node contains only n…

oracle之单行函数之分组函数之课后练习

33. 查询 employees 表中有多少个部门select count(distinct department_id)from employees 34. 查询全公司奖金基数的平均值(没有奖金的人按 0 计算)select avg(nvl(commission_pct, 0))from employees 35. 查询各个部门的平均工资--错误: avg(salary) 返回公司平均工资, 只…

谭浩强c语言规范化的指数形式,C语言程序设计谭浩强第四期末复习重点.docx

1.1.问题分析2.设计算法3.编写程序4.对源程序进行编辑、编译和连接5.运行程序&#xff0c;分析结 6.编写程序文档第一章程 序 设 计 和C 语 言1.1.什么是计算机程序程序&#xff1a;一组计算机能识别和执行的指令。只要让计算机执行这个程序&#xff0c;计算机就会自动地、有条…

OPT和LRU页面置换算法C语言代码,页面置换算法模拟——OPT、FIFO和LRU算法.doc

实用标准文案精彩文档操作系统实验报告页面置换算法模拟——OFT、FIFO和LRU算法班级&#xff1a;2013级软件工程1班学号&#xff1a;X X X姓名&#xff1a;萧氏一郎数据结构说明&#xff1a;Memery[10]物理块中的页码Page[100]页面号引用串Temp[100][10]辅助数组Void print(uns…

使用Enterprise Architecture绘制10种UML画画

UML绘制10种课程要求UML画画&#xff0c;选Enterprise Architecture作为一个绘图工具&#xff0c;每一个草图必须是网上找教程&#xff0c;我觉得很麻烦&#xff0c;还有一些数字并没有找到详细的教程。在我自己找一个绘图方法&#xff0c;今天总结使用Enterprise Architecture…

RocketMQ初步应用架构理论

RocketMQ初步应用架构理论 写给RocketMQ架构应用入门&#xff0c;内容涉及它的设计机理以及推到出来的应用注意事项&#xff0c;入门人员请看。 稍微涉及技术细节&#xff0c;留以我设计中间件时参考&#xff0c;将来整理深度文档时会抽取走&#xff0c;入门人员可以无视。 以下…

android程序的入口点,常见android面试基础题

2015-11-21 06:30:02阅读( 1344 )1. Intent的几种有关Activity启动的方式有哪些&#xff0c;你了解每个含义吗这里Android123提示大家&#xff0c;Intent的一些标记有FLAG_ACTIVITY_BROUGHT_TO_FRONT、FLAG_ACTIVITY_CLEAR_TOP、FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET、FLAG_ACT…

转:pysqlite笔记

这是一篇老笔记&#xff0c;原来是放在旧博客上 的&#xff0c;最近因为公司内部一个小东西&#xff0c;想使用简单点的数据库来存储数据&#xff0c;就想起用SQLite来做&#xff0c;上网搜索一些教程。竟然发现&#xff0c;原来一年多前&#xff0c;我也学过一阵 子&#xff0…

android页面布局更改,使用setContentView的方式更换布局文件从而更换界面

使用转换Activity的布局文件的方式&#xff0c;从而达到转换android页面的目的(这里没有使用Intent)&#xff1a;程序很简单&#xff0c;摆一个大概出来&#xff1a;package com.seed.lee.setContentView;import android.app.Activity;import android.os.Bundle;import android…

oracle之单行函数之子查询

--睡得工资比abel高 select last_name,salary from employees where salary>(select salary from employees where last_nameAbel)运行结果 --返回job_id 与141号员工相同 salary比143号多 select last_name,job_id,salary from employees where job_id(select job_id from…

android模拟gps定位软件,gps定位模拟器下载最新版

卫星地图导航&#xff0c;从此出门想去哪里去哪里&#xff0c;再也不用因为不知道路线而烦忧&#xff01;它还能实时定位&#xff0c;快速找人、找车&#xff01;推荐&#xff01;使用前提&#xff1a;1、定位模拟器是基于Xposed安卓框架下的插件&#xff0c;因此安装定位模拟器…

Java的堆与栈,科普给大家

1. 栈(stack)与堆(heap)都是Java用来在Ram中存放数据的地方。与C不同&#xff0c;Java自动管理栈和堆&#xff0c;程序员不能直接地设置栈或堆。 2. 栈的优势是&#xff0c;存取速度比堆要快&#xff0c;仅次于直接位于CPU中的寄存器。但缺点是&#xff0c;存在栈中的数据大小与…

android edittext禁止输入特殊字符,Android EditText禁止输入空格和特殊字符

/*** 禁止EditText输入特殊字符* param editText*/public static void setEditTextInhibitInputSpeChat(EditText editText){InputFilter filternew InputFilter() {Overridepublic CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, i…

oracle之单行函数之子查询之课后练习

/*************************************************************************************************/ 40. 谁的工资比 Abel 高?1). 写两条 SQL 语句.SELECT salaryFROM employeesWHERE last_name Abel--返回值为 11000SELECT last_name, salaryFROM employeesWHERE sal…

移动端开发的知识系统介绍

移动端开发1. 移动端适配&#xff1a;http://suqing.iteye.com/blog/1982733http://www.douban.com/note/261319445/ http://www.woshipm.com/ucd/150207.html<meta name"screen-orientation" content"portrait"><!-- 强制竖屏 --><meta na…

delphi android动态权限,DELPHI安卓动态权限申请

DELPHI安卓动态权限申请安卓8.0以前的版本&#xff0c;只需要给静态权限就可以了&#xff0c;但安卓8.0及以后的版本&#xff0c;还需要运行期用代码动态申请权限。下面以《蓝牙权限》为例&#xff0c;其他权限类似。Delphi 10.3 社区版&#xff0c;提供的 Sample 里面有一个例…