- 浏览: 290075 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
86614009:
如何在service层,如何获取绑定到当前线程的entitna ...
使用spring的OpenEntityManagerInView -
yajunyajun2011:
好帖子 怎么没人顶呢
Java 正则表达式最大,最小匹配问题 -
xtuali:
能说明一下,你的nutch是哪个版本的吗?谢谢!
搜索引擎Nutch源代码研究之一 网页抓取(1) -
dongmusic:
需要学习这么多的东西,吐血中...
如何提高Java开发能力 -
jiminsc:
cool
LDAP 验证、添加、修改、删除(转)
文章分类:Java编程
1. 代理模式
代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
代理模式一般涉及到的角色有:
抽象角色:声明真实对象和代理对象的共同接口;
代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装。
真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。
由以上代码可以看出,客户实际需要调用的是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接口,以第一节中的示例为例:
抽象角色(之前是抽象类,此处应改为接口):
具体角色RealSubject:同上;
该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;此外,在该类还实现了invoke方法,该方法中的
method.invoke(sub,args);
其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,args为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。
通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject类)也可以动态改变,从而实现了非常灵活的动态代理关系。
以上是转载别人的,一个转载人的理解:
在使用动态代理的时候通过Proxy动态的生成一个代理对象,该代理对象与被代理的对象享有相同的函数接口。使用这种动态代理可以自动的截获被代理类的各种方法,并通过代理类在动作执行的前后添加不同的预处理动作。
生成代理对象的方法通过
newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)
参数分别说明,1.被代理对象的ClassLoder,2.被代理对象的接口组,3.实现代理invoke接口的代理对象,在loader的任何操作被执行的时候调用h的具体方法进行改造。注意newProxyInstance是一个object该对象与loader的实际类型相同。
代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
代理模式一般涉及到的角色有:
抽象角色:声明真实对象和代理对象的共同接口;
代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装。
真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。
- 抽象角色:
- 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();
- 抽象角色:
- 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();
抽象角色: 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接口,以第一节中的示例为例:
抽象角色(之前是抽象类,此处应改为接口):
public interface Subject{ abstract public void request(); }
具体角色RealSubject:同上;
- 代理角色:
- 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;
- }
- }
- 代理角色:
- 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;
- }
- }
代理角色: 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为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。
- 客户端:
- 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();
- }
- 客户端:
- 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();
- }
客户端: 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处理就可以。不用像代理模式中写出相应的方法。
发表评论
-
哈希表的ELFhash算法
2011-12-09 10:25 1384算法: while(*key)//遍历字符串 ... -
有理想的程序员必须知道的15件事(转自CSDN)
2011-07-21 21:23 603作为程序员,要取得非凡成就需要记住的15件事。 1.走一条不 ... -
每天写出好代码的5个建议(转自CSDN)
2011-07-21 21:18 573成为一个优秀的程序员 ... -
程序员技术练级攻略(转自CSDN)
2011-07-21 21:05 1033导读:本文是由陈皓和 ... -
学android开发 不得不去的好地方
2011-07-06 17:08 864中国移动开发者社区 h ... -
ppt制作
2011-04-04 18:50 938PowerPoint的动画效果比较多,但图片只能一幅一幅地动作 ... -
Java 虚拟机体系结构
2011-02-22 10:11 827众所周知,Java源代码被编译器编译成class文件。而并 ... -
提高程序运行效率的方法
2011-02-22 09:36 831浏览器发送一次请求 ... -
从JVM内存管理的角度谈谈静态方法和静态属性
2011-02-22 09:27 677作者 robbin (htt ... -
模型驱动软件开发实战步骤
2011-01-14 15:31 786有人说:今年是AJ ... -
实战DDD(Domain-Driven Design领域驱动设计)
2011-01-14 15:30 8802004年著名建模专家 ... -
领域模型驱动设计(DDD)之模型提炼
2011-01-14 15:30 829当Java世界提供的可 ... -
模型驱动设计(MDD)之灵活设计
2011-01-14 15:29 658灵活设计可以使我 ... -
JAVA开发者应该去的20个英文网站
2011-01-14 10:35 638http://www.javaalmanac. ... -
Java httpclient解决方案中的中文传递
2010-12-27 09:39 22281 Commons HttpClient 开源项目简介 Htt ... -
java字符串的各种编码转换
2010-12-26 19:57 894import java.io.UnsupportedEncod ... -
firebug的使用
2010-12-26 11:13 816什么是Firebug 从事了数 ... -
HttpClient 学习整理
2010-12-13 09:32 788HTTP 协议可能是现在 Internet 上使用得最多、 ... -
JAVA使用java.net.url模拟登录
2010-12-08 13:32 1484有的时候,会需要使用java的程序访问网页,正常的访问网页的程 ... -
面试知识点
2010-12-06 10:54 1095总结了一些要弄懂的知识点:JVM的构造框架,TCP的三次握手, ...
相关推荐
基于改进YOLOv5s的森林烟火检测算法.pdf
人力资源管理工具绩效考核excel模板01
施工班组长绩效考核表
57 -营业部经理绩效考核表1
XX公司行政部绩效考核指标
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)
11-6-质检员绩效考核表(含自动计算、等级评价及任意设置等级)
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成为了许多公共项目、研究机构和商业公司的重要数据来源。
部门绩效考核评分表
12-11-运输车队长绩效考核表(含自动计算、等级评价)
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方法
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个月以内的入境游客抽样调查资料,由综合分析报告和调查分类数据两部分组成,分类数据包括:入境游客的主要特征,入境外国人、港澳台同胞的花费水平和花费构成、在境内的停留时间以及入境次数、流向和对住宿单位的选择等。 下篇为国内旅游抽样调查资料,汇集了对城镇居民和农村居民的国内旅游抽样调查结果,共分为四个部分:第一部分为综合分析报告;第二部分为国内旅游出游及花费情况;第三部分为城镇居民国内旅游抽样调查分类数据;第四部分为农村居民国内旅游抽样调查分类数据。
1、表单界面,身份证信息保存在dbf表中,供vfp应用使用,可导出为xls电子表格。 2、提供了身份证过期校验和查询功能。
人事行政主管绩效考核评分表
08 -大堂副理绩效考核表1
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