`

对代理模式与Java动态代理类的理解(三转)

阅读更多
文章分类:Java编程
1. 代理模式
代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
代理模式一般涉及到的角色有:
抽象角色:声明真实对象和代理对象的共同接口;
代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装。
真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。
Java代码 复制代码
  1. 抽象角色:   
  2. abstract public class Subject{   
  3. abstract public void request();   
  4. }   
  5. 真实角色:实现了Subject的request()方法。   
  6. public class RealSubject extends Subject{   
  7. public RealSubject(){   
  8. }   
  9. public void request() {   
  10. System.out.println("From real subject.");   
  11. }   
  12. }   
  13. 代理角色:   
  14. public class ProxySubject extends Subject {   
  15. private RealSubject realSubject; //以真实角色作为代理角色的属性   
  16. public ProxySubject() { }   
  17. public void request(){ //该方法封装了真实对象的request方法   
  18. preRequest();   
  19. if( realSubject == null ){   
  20. realSubject = new RealSubject();   
  21. }   
  22. realSubject.request(); //此处执行真实对象的request方法   
  23. postRequest();   
  24. }   
  25. private void preRequest(){   
  26. //something you want to do before requesting   
  27. }   
  28. private void postRequest(){   
  29. //something you want to do after requesting   
  30. }   
  31. }   
  32. 客户端调用:   
  33. Subject sub=new ProxySubject();   
  34. Sub.request();  
Java代码 复制代码
  1. 抽象角色:   
  2. abstract public class Subject{   
  3. abstract public void request();   
  4. }   
  5. 真实角色:实现了Subject的request()方法。   
  6. public class RealSubject extends Subject{   
  7. public RealSubject(){   
  8. }   
  9. public void request() {   
  10. System.out.println("From real subject.");   
  11. }   
  12. }   
  13. 代理角色:   
  14. public class ProxySubject extends Subject {   
  15. private RealSubject realSubject; //以真实角色作为代理角色的属性   
  16. public ProxySubject() { }   
  17. public void request(){ //该方法封装了真实对象的request方法   
  18. preRequest();   
  19. if( realSubject == null ){   
  20. realSubject = new RealSubject();   
  21. }   
  22. realSubject.request(); //此处执行真实对象的request方法   
  23. postRequest();   
  24. }   
  25. private void preRequest(){   
  26. //something you want to do before requesting   
  27. }   
  28. private void postRequest(){   
  29. //something you want to do after requesting   
  30. }   
  31. }   
  32. 客户端调用:   
  33. Subject sub=new ProxySubject();   
  34. Sub.request();  
抽象角色:
abstract public class Subject{
abstract public void request();
}
真实角色:实现了Subject的request()方法。
public class RealSubject extends Subject{
public RealSubject(){
}
public void request() {
System.out.println("From real subject.");
}
}
代理角色:
public class ProxySubject extends Subject {
private RealSubject realSubject; //以真实角色作为代理角色的属性
public ProxySubject() { }
public void request(){ //该方法封装了真实对象的request方法
preRequest();
if( realSubject == null ){
realSubject = new RealSubject();
}
realSubject.request(); //此处执行真实对象的request方法
postRequest();
}
private void preRequest(){
//something you want to do before requesting
}
private void postRequest(){
//something you want to do after requesting
}
}
客户端调用:
Subject sub=new ProxySubject();
Sub.request();

由以上代码可以看出,客户实际需要调用的是RealSubject类的request()方法,现在用ProxySubject来代理RealSubject类,同样达到目的,同时还封装了其他方法(preRequest(),postRequest()),可以处理一些其他问题。
另外,如果要按照上述的方法使用代理模式,那么真实角色必须是事先已经存在的,并将其作为代理对象的内部属性。但是实际使用时,一个真实角色必须对应一个代理角色,如果大量使用会导致类的急剧膨胀;此外,如果事先并不知道真实角色,该如何使用代理呢?这个问题可以通过Java的动态代理类来解决。
2.动态代理类 (注,重点是这个)
Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类:
(1). Interface InvocationHandler:该接口中仅定义了一个方法Object:invoke(Object obj,Method method, Object[] args)。在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的request(),args为该方法的参数数组。这个抽象方法在代理类中动态实现。
(2).Proxy:该类即为动态代理类,作用类似于上例中的ProxySubject,其中主要包含以下内容:
Protected Proxy(InvocationHandler h):构造函数,估计用于给内部的h赋值。
Static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。
Static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法)。
所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然啦,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作
在使用动态代理类时,我们必须实现InvocationHandler接口,以第一节中的示例为例:
抽象角色(之前是抽象类,此处应改为接口):

Java代码 复制代码
  1. public interface Subject{   
  2. abstract public void request();   
  3. }  
Java代码 复制代码
  1. public interface Subject{   
  2. abstract public void request();   
  3. }  
public interface Subject{
abstract public void request();
}


具体角色RealSubject:同上;
Java代码 复制代码
  1. 代理角色:   
  2. import java.lang.reflect.Method;   
  3. import java.lang.reflect.InvocationHandler;   
  4. public class DynamicSubject implements InvocationHandler {   
  5. private Object sub;   
  6. public DynamicSubject() {   
  7. }   
  8. public DynamicSubject(Object obj) {   
  9. sub = obj;   
  10. }   
  11. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {   
  12. System.out.println("before calling " + method);   
  13. method.invoke(sub,args);   
  14. System.out.println("after calling " + method);   
  15. return null;   
  16. }   
  17. }  
Java代码 复制代码
  1. 代理角色:   
  2. import java.lang.reflect.Method;   
  3. import java.lang.reflect.InvocationHandler;   
  4. public class DynamicSubject implements InvocationHandler {   
  5. private Object sub;   
  6. public DynamicSubject() {   
  7. }   
  8. public DynamicSubject(Object obj) {   
  9. sub = obj;   
  10. }   
  11. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {   
  12. System.out.println("before calling " + method);   
  13. method.invoke(sub,args);   
  14. System.out.println("after calling " + method);   
  15. return null;   
  16. }   
  17. }  
代理角色:
import java.lang.reflect.Method;
import java.lang.reflect.InvocationHandler;
public class DynamicSubject implements InvocationHandler {
private Object sub;
public DynamicSubject() {
}
public DynamicSubject(Object obj) {
sub = obj;
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("before calling " + method);
method.invoke(sub,args);
System.out.println("after calling " + method);
return null;
}
}

该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;此外,在该类还实现了invoke方法,该方法中的
method.invoke(sub,args);
其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,args为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。
Java代码 复制代码
  1. 客户端:   
  2. import java.lang.reflect.InvocationHandler;   
  3. import java.lang.reflect.Proxy;   
  4. import java.lang.reflect.Constructor;   
  5. import java.lang.reflect.Method;   
  6. public class Client{   
  7. static public void main(String[] args) throws Throwable{   
  8. RealSubject rs = new RealSubject(); //在这里指定被代理类   
  9. InvocationHandler ds = new DynamicSubject(rs); //初始化代理类   
  10. Class cls = rs.getClass();   
  11. //以下是分解步骤   
  12. /*  
  13. Class c = Proxy.getProxyClass(cls.getClassLoader(),cls.getInterfaces()) ;  
  14. Constructor ct=c.getConstructor(new Class[]{InvocationHandler.class});  
  15. Subject subject =(Subject) ct.newInstance(new Object[]{ds});  
  16. */  
  17. //以下是一次性生成   
  18. Subject subject = (Subject) Proxy.newProxyInstance(cls.getClassLoader(),   
  19. cls.getInterfaces(),ds );   
  20. subject.request();   
  21. }  
Java代码 复制代码
  1. 客户端:   
  2. import java.lang.reflect.InvocationHandler;   
  3. import java.lang.reflect.Proxy;   
  4. import java.lang.reflect.Constructor;   
  5. import java.lang.reflect.Method;   
  6. public class Client{   
  7. static public void main(String[] args) throws Throwable{   
  8. RealSubject rs = new RealSubject(); //在这里指定被代理类   
  9. InvocationHandler ds = new DynamicSubject(rs); //初始化代理类   
  10. Class cls = rs.getClass();   
  11. //以下是分解步骤   
  12. /*  
  13. Class c = Proxy.getProxyClass(cls.getClassLoader(),cls.getInterfaces()) ;  
  14. Constructor ct=c.getConstructor(new Class[]{InvocationHandler.class});  
  15. Subject subject =(Subject) ct.newInstance(new Object[]{ds});  
  16. */  
  17. //以下是一次性生成   
  18. Subject subject = (Subject) Proxy.newProxyInstance(cls.getClassLoader(),   
  19. cls.getInterfaces(),ds );   
  20. subject.request();   
  21. }  
客户端:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
public class Client{
static public void main(String[] args) throws Throwable{
RealSubject rs = new RealSubject(); //在这里指定被代理类
InvocationHandler ds = new DynamicSubject(rs); //初始化代理类
Class cls = rs.getClass();
//以下是分解步骤
/*
Class c = Proxy.getProxyClass(cls.getClassLoader(),cls.getInterfaces()) ;
Constructor ct=c.getConstructor(new Class[]{InvocationHandler.class});
Subject subject =(Subject) ct.newInstance(new Object[]{ds});
*/
//以下是一次性生成
Subject subject = (Subject) Proxy.newProxyInstance(cls.getClassLoader(),
cls.getInterfaces(),ds );
subject.request();
}

通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject类)也可以动态改变,从而实现了非常灵活的动态代理关系。
以上是转载别人的,一个转载人的理解:
在使用动态代理的时候通过Proxy动态的生成一个代理对象,该代理对象与被代理的对象享有相同的函数接口。使用这种动态代理可以自动的截获被代理类的各种方法,并通过代理类在动作执行的前后添加不同的预处理动作。
生成代理对象的方法通过
newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)
参数分别说明,1.被代理对象的ClassLoder,2.被代理对象的接口组,3.实现代理invoke接口的代理对象,在loader的任何操作被执行的时候调用h的具体方法进行改造。注意newProxyInstance是一个object该对象与loader的实际类型相同。
自己的理解:代理结构
interface---->instance                handler(instance)处理程序
                         |                                  |
                        -------------------------------
                                         |
                          代理类(interface类型)
这么看就是中介 代理的作用。动态是因为调用所有的instance的方法,都只用handler的invoke处理就可以。不用像代理模式中写出相应的方法。
分享到:
评论

相关推荐

    基于改进YOLOv5s的森林烟火检测算法.pdf

    基于改进YOLOv5s的森林烟火检测算法.pdf

    人力资源管理工具绩效考核excel模板01.xlsx

    人力资源管理工具绩效考核excel模板01

    施工班组长绩效考核表.xls

    施工班组长绩效考核表

    57 -营业部经理绩效考核表1.xlsx

    57 -营业部经理绩效考核表1

    XX公司行政部绩效考核指标.xls

    XX公司行政部绩效考核指标

    ant-apache-xalan2-1.9.4-2.el7.x64-86.rpm.tar.gz

    1、文件内容:ant-apache-xalan2-1.9.4-2.el7.rpm以及相关依赖 2、文件形式:tar.gz压缩包 3、安装指令: #Step1、解压 tar -zxvf /mnt/data/output/ant-apache-xalan2-1.9.4-2.el7.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm 4、安装指导:私信博主,全程指导安装

    部门绩效考核表模板(基于KPI以月度为例2).xlsx

    部门绩效考核表模板(基于KPI以月度为例2)

    11-6-质检员绩效考核表(含自动计算、等级评价及任意设置等级).xlsx

    11-6-质检员绩效考核表(含自动计算、等级评价及任意设置等级)

    2024年最新全国河流、湖泊矢量数据(数据权威)

    2024最新全国河流湖泊矢量数据 【数据介绍】 2024年中国河流湖泊数据 一份包含中国境内所有主要河流和湖泊的地理信息数据。 数据格式:Shapefile:广泛使用的GIS数据格式,方便在各类GIS软件中使用。 数据获取:访问OpenStreetMap官网,通过导出工具选择中国区域并下载所需的数据。 使用Geofabrik等第三方网站,可以下载预处理好的中国区域的OSM数据。 数据使用:GIS软件:如QGIS、ArcGIS等,用户可以在这些软件中导入OSM数据进行可视化、分析和编辑。 数据应用: 环境研究:分析河流湖泊的水质变化,研究水资源分布及其环境影响。 城市规划:用于规划城市水系、洪水防控、水资源管理等。 导航和旅游:为河流湖泊的导航和旅游路线规划提供数据支持。 科研:为水文地理研究、生态保护、气候变化等领域提供基础数据。 数据特点: 实时更新:OSM数据由全球用户贡献,具有较高的实时性和更新频率。 开放性:所有数据都在开放许可下发布,允许用户自由使用、修改和分发。 详细性:由于全球志愿者的不断努力,数据细节较为丰富,涵盖了从主要河流湖泊到小型水体的广泛范围。 数据时间2024年5月,shp格式,数据来源OpenStreetMap。 OpenStreetMap(OSM)介绍: 一个开放的、免费的、全球性的地图项目,由全球的志愿者和地图爱好者们共同创建和维护。 OSM的数据包括道路、建筑、公园、河流、湖泊等各类地理信息。由于是由众多志愿者共同编辑,OSM的数据具有很高的实时性和详细程度,特别是在一些活跃的区域,地图数据的更新速度和精度往往超过商业地图服务。 用户可以直接在OSM官网下载地图数据,数据格式主要有OSM XML和PBF等。此外,还有一些第三方网站和工具提供更加便捷的数据下载和处理服务,如Geofabrik、Overpass API等。 OSM的数据可以在各种GIS软件中使用,如QGIS、ArcGIS等。此外,还可以使用Python的OSMnx、GeoPandas等库进行编程处理,或者通过Leaflet、Mapbox等JavaScript库将OSM数据集成到web地图应用中。 OSM的所有数据都在开放许可下发布,允许用户自由使用、修改和分发。这使得OSM成为了许多公共项目、研究机构和商业公司的重要数据来源。

    部门绩效考核评分表.xlsx

    部门绩效考核评分表

    12-11-运输车队长绩效考核表(含自动计算、等级评价).xlsx

    12-11-运输车队长绩效考核表(含自动计算、等级评价)

    ant-javadoc-1.9.4-2.el7.x64-86.rpm.tar.gz

    1、文件内容:ant-javadoc-1.9.4-2.el7.rpm以及相关依赖 2、文件形式:tar.gz压缩包 3、安装指令: #Step1、解压 tar -zxvf /mnt/data/output/ant-javadoc-1.9.4-2.el7.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm 4、安装指导:私信博主,全程指导安装

    springboot整合 freemarker方法

    springboot整合 freemarker方法

    apache-commons-codec-1.8-7.el7.x64-86.rpm.tar.gz

    1、文件内容:apache-commons-codec-1.8-7.el7.rpm以及相关依赖 2、文件形式:tar.gz压缩包 3、安装指令: #Step1、解压 tar -zxvf /mnt/data/output/apache-commons-codec-1.8-7.el7.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm 4、安装指导:私信博主,全程指导安装

    (数据权威)全国旅游抽样调查数据

    《旅游抽样调查资料》是反映入境游客在华(内地)花费和国内居民国内旅游情况的资料性年刊,分为上下两篇。 上篇为在华(内地)停留时间在3个月以内的入境游客抽样调查资料,由综合分析报告和调查分类数据两部分组成,分类数据包括:入境游客的主要特征,入境外国人、港澳台同胞的花费水平和花费构成、在境内的停留时间以及入境次数、流向和对住宿单位的选择等。 下篇为国内旅游抽样调查资料,汇集了对城镇居民和农村居民的国内旅游抽样调查结果,共分为四个部分:第一部分为综合分析报告;第二部分为国内旅游出游及花费情况;第三部分为城镇居民国内旅游抽样调查分类数据;第四部分为农村居民国内旅游抽样调查分类数据。

    二代身份证信息读取(vfp8.0)

    1、表单界面,身份证信息保存在dbf表中,供vfp应用使用,可导出为xls电子表格。 2、提供了身份证过期校验和查询功能。

    人事行政主管绩效考核评分表.xls

    人事行政主管绩效考核评分表

    08 -大堂副理绩效考核表1.xlsx

    08 -大堂副理绩效考核表1

    apr-1.4.8-7.el7.x64-86.rpm.tar.gz

    1、文件内容:apr-1.4.8-7.el7.rpm以及相关依赖 2、文件形式:tar.gz压缩包 3、安装指令: #Step1、解压 tar -zxvf /mnt/data/output/apr-1.4.8-7.el7.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm 4、安装指导:私信博主,全程指导安装

    ComponentNameError解决办法.md

    ComponentNameError解决办法.md

Global site tag (gtag.js) - Google Analytics