【转】java的反射机制中的动态代理代理(一)--简介

1. 代理模式

代理模式是常用的Java设计模式,它的特征是代理类与委托类有同样的接口,如下图所示。代理类主要负责为委托类预处理消息过滤消息把消息转发给委托类以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

http://new.51cto.com/files/uploadimg/20070211/1712020.jpg

按照代理类的创建时期,代理类可分为两种。

◆静态代理类:由程序员创建或由特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。

◆动态代理类:在程序运行时,运用反射机制动态创建而成。

2.动态代理类

与静态代理类对照的是动态代理类,动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java反射机制可以生成任意类型的动态代理类。java.lang.reflect包中的Proxy类和 InvocationHandler接口提供了生成动态代理类的能力。

2.1 Proxy类

Proxy类提供了创建动态代理类及其实例的静态方法。

(1)getProxyClass()静态方法负责创建动态代理类,它的完整定义如下:

public static Class getProxyClass(ClassLoader loader, Class[] interfaces)
                                 throws IllegalArgumentException

参数loader指定动态代理类的类加载器,参数interfaces指定动态代理类需要实现的所有接口。

(2)newProxyInstance()静态方法负责创建动态代理类的实例,它的完整定义如下:

public static Object newProxyInstance(ClassLoader loader,
Class[] interfaces,
InvocationHandler handler)
throws IllegalArgumentException

参数loader指定动态代理类的类加载器,参数interfaces指定动态代理类需要实现的所有接口,参数handler指定与动态代理类关联的 InvocationHandler对象。

举例,以下两种方式都创建了实现Foo接口的动态代理类的实例:

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接口的动态代理类的实例。

http://new.51cto.com/files/uploadimg/20070212/105519704.jpg

例  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://new.51cto.com/files/uploadimg/20070212/105539180.jpg

转自:http://developer.51cto.com/art/200702/40215.htmJava网络编程精解

posted @ 2011-05-23 21:42  vivianC  阅读(851)  评论(1编辑  收藏  举报