【转】java的反射机制中的动态代理代理(一)--简介
2011-05-23 21:42
841 查看
1. 代理模式
代理模式是常用的Java设计模式,它的特征是代理类与委托类有同样的接口,如下图所示。代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。
View Code
由Proxy类的静态方法创建的动态代理类具有以下特点:
◆动态代理类是public、final和非抽象类型的;
◆动态代理类继承了java.lang.reflect.Proxy类;
◆动态代理类的名字以“$Proxy”开头;
◆动态代理类实现getProxyClass()和newProxyInstance()方法中参数interfaces指定的所有接口;
◆Proxy类的isProxyClass(Class cl)静态方法可用来判断参数指定的类是否为动态代理类。只有通过Proxy类创建的类才是动态代理类;
◆动态代理类都具有一个public类型的构造方法,该构造方法有一个InvocationHandler类型的参数。
由Proxy类的静态方法创建的动态代理类的实例具有以下特点:
◆假定变量foo是一个动态代理类的实例,并且这个动态代理类实现了Foo接口,那么“foo instanceof Foo”的值为true。把变量foo强制转换为Foo类型是合法的:(Foo) foo //合法
◆每个动态代理类实例都和一个InvocationHandler实例关联。Proxy类的getInvocationHandler(Object proxy)静态方法返回与参数proxy指定的代理类实例所关联的InvocationHandler对象。
◆假定Foo接口有一个amethod()方法,那么当程序调用动态代理类实例foo的amethod()方法时,该方法会调用与它关联的InvocationHandler对象的invoke()方法。
2.2 InvocationHandler接口
InvocationHandler接口为方法调用接口,它声明了负责调用任意一个方法的invoke()方法:
参数proxy指定动态代理类实例,参数method指定被调用的方法,参数args指定向被调用方法传递的参数,invoke()方法的返回值表示被调用方法的返回值。
2.3 举例
如下图所示,HelloServiceProxyFactory类(如例程10-15所示)的getHello- ServiceProxy()静态方法负责创建实现了HelloService接口的动态代理类的实例。
例 HelloServiceProxyFactory.java
如下所示的Client2类先创建了一个HelloServiceImpl实例,然后创建了一个动态代理类实例helloServiceProxy,最后调用动态代理类实例的echo()方法。
打印结果如下:
转自:http://developer.51cto.com/art/200702/40215.htm 《Java网络编程精解》
代理模式是常用的Java设计模式,它的特征是代理类与委托类有同样的接口,如下图所示。代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。
View Code
/**** 方式一 ****/ //创建InvocationHandler对象 InvocationHandler handler = new MyInvocationHandler(...); //创建动态代理类 Class proxyClass = Proxy.getProxyClass( Foo.class.getClassLoader(), new Class[] { Foo.class }); //创建动态代理类的实例 Foo foo = (Foo) proxyClass.getConstructor( new Class[] { InvocationHandler.class}). newInstance(new Object[] { handler }); /**** 方式二 ****/ //创建InvocationHandler对象 InvocationHandler handler = new MyInvocationHandler(...); //直接创建动态代理类的实例 Foo foo = (Foo) Proxy.newProxyInstance(Foo.class.getClassLoader(), new Class[] { Foo.class }, handler);
由Proxy类的静态方法创建的动态代理类具有以下特点:
◆动态代理类是public、final和非抽象类型的;
◆动态代理类继承了java.lang.reflect.Proxy类;
◆动态代理类的名字以“$Proxy”开头;
◆动态代理类实现getProxyClass()和newProxyInstance()方法中参数interfaces指定的所有接口;
◆Proxy类的isProxyClass(Class cl)静态方法可用来判断参数指定的类是否为动态代理类。只有通过Proxy类创建的类才是动态代理类;
◆动态代理类都具有一个public类型的构造方法,该构造方法有一个InvocationHandler类型的参数。
由Proxy类的静态方法创建的动态代理类的实例具有以下特点:
◆假定变量foo是一个动态代理类的实例,并且这个动态代理类实现了Foo接口,那么“foo instanceof Foo”的值为true。把变量foo强制转换为Foo类型是合法的:(Foo) foo //合法
◆每个动态代理类实例都和一个InvocationHandler实例关联。Proxy类的getInvocationHandler(Object proxy)静态方法返回与参数proxy指定的代理类实例所关联的InvocationHandler对象。
◆假定Foo接口有一个amethod()方法,那么当程序调用动态代理类实例foo的amethod()方法时,该方法会调用与它关联的InvocationHandler对象的invoke()方法。
2.2 InvocationHandler接口
InvocationHandler接口为方法调用接口,它声明了负责调用任意一个方法的invoke()方法:
Object invoke(Object proxy,Method method,Object[] args) throws Throwable
参数proxy指定动态代理类实例,参数method指定被调用的方法,参数args指定向被调用方法传递的参数,invoke()方法的返回值表示被调用方法的返回值。
2.3 举例
如下图所示,HelloServiceProxyFactory类(如例程10-15所示)的getHello- ServiceProxy()静态方法负责创建实现了HelloService接口的动态代理类的实例。
例 HelloServiceProxyFactory.java
package proxy; import java.lang.reflect.*; public class HelloServiceProxyFactory { /** 创建一个实现了HelloService接口的动态代理类的实例 * 参数helloService引用被代理的HelloService实例 */ public static HelloService getHelloServiceProxy(final HelloService helloService){ //创建一个实现了InvocationHandler接口的匿名类的实例 InvocationHandler handler=new InvocationHandler(){ public Object invoke(Object proxy, Method method, Object args[]) throws Exception{ System.out.println("before calling "+method); //预处理 Object result=method.invoke(helloService,args); //调用被代理的HelloService实例的方法 System.out.println("after calling "+method); //事后处理 return result; } }; Class classType=HelloService.class; return (HelloService)Proxy.newProxyInstance(classType.getClassLoader(), new Class[]{classType}, handler); }//# getHelloServiceProxy() }
如下所示的Client2类先创建了一个HelloServiceImpl实例,然后创建了一个动态代理类实例helloServiceProxy,最后调用动态代理类实例的echo()方法。
package proxy; public class Client2{ public static void main(String args[]){ HelloService helloService=new HelloServiceImpl(); HelloService helloServiceProxy= HelloServiceProxyFactory.getHelloServiceProxy(helloService); System.out.println("动态代理类的名字为" +helloServiceProxy.getClass().getName()); System.out.println(helloServiceProxy.echo("Hello")); } }
打印结果如下:
动态代理类的名字为$Proxy0 before calling public abstract java.lang. String proxy.HelloService.echo(java.lang.String) after calling public abstract java.lang. String proxy.HelloService.echo(java.lang.String) echo:Hello
转自:http://developer.51cto.com/art/200702/40215.htm 《Java网络编程精解》
相关文章推荐
- Java学习之反射机制---动态代理
- java动态代理机制和反射机制间的联系
- java.lang.Class<T> -- 反射机制及动态代理
- java学习之路 之 反射机制综合练习题、动态代理实例
- JAVA中的反射机制和动态代理
- JAVA的反射机制与动态代理
- java_IO,反射机制,jdk动态代理
- java学习笔记---类型信息(type information)、反射机制与动态代理
- 关于java的反射机制及动态代理
- 深入Java机制(一)--反射机制和动态代理机制
- Java学习笔记:反射与代理机制(静态、动态)
- 模拟实现Struts拦截器-蕴含着代理模式,AOP,工厂模式,依赖注入,Java 反射,动态构造等机制
- Java的反射机制与动态代理(二):动态代理
- 【转】java的反射机制中的动态代理代理(二)--在远程方法调用中运用代理类
- Java 动态代理与反射机制
- Java的反射机制与动态代理(一):反射
- Java的反射机制与动态代理学习笔记
- Java中的动态代理及反射机制
- Java的反射机制和动态代理
- JAVA 反射机制与动态代理