java 代理(proxy)模式

代理模式(Proxy Pattern)

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

 

如下列子:

[java]  view plain copy
  1. package proxy;  
  2.   
  3. interface Dao{  
  4.     public void insert();  
  5. }  
  6.   
  7. class JdbcDao implements Dao{  
  8.     public void insert(){  
  9.         System.out.println("in jdbc insert");  
  10.     }  
  11. }  
  12. class HibernateDao implements Dao{  
  13.     public void insert(){  
  14.         System.out.println("in hibernate insert");  
  15.     }  
  16. }  
  17.   
  18. class ProxyDao implements Dao{  
  19.     private Dao dao;  
  20.     public ProxyDao(Dao dao){  
  21.         this.dao=dao;  
  22.     }  
  23.     public void insert() {  
  24.         System.out.println("write log before invoke");  
  25.         dao.insert();  
  26.         System.out.println("write log after invoke");  
  27.     }  
  28. }  
  29.   
  30. public class Test {  
  31.     public static void main(String[] args) {  
  32.         Dao jdbcDao =new JdbcDao();  
  33.         Dao proxydao = new ProxyDao(jdbcDao);  
  34.         proxydao.insert();  
  35.     }  
  36. }  

 

运行结果:

 

write log before invoke
in jdbc insert
write log after invoke

 

    随着Proxy的流行,Sun把它纳入到JDK1.3实现了Java的动态代理。动态代理和普通的代理模式的区别,就是动态代理中的代理类是由 java.lang.reflect.Proxy类在运行期时根据接口定义,采用Java反射功能动态生成的。和 java.lang.reflect.InvocationHandler结合,可以加强现有类的方法实现。如图2,图中的自定义Handler实现 InvocationHandler接口,自定义Handler实例化时,将实现类传入自定义Handler对象。自定义Handler需要实现 invoke方法,该方法可以使用Java反射调用实现类的实现的方法,同时当然可以实现其他功能,例如在调用实现类方法前后加入Log。而Proxy类根据Handler和需要代理的接口动态生成一个接口实现类的对象。当用户调用这个动态生成的实现类时,实际上是调用了自定义Handler的 invoke方法。

 

 

 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接口的动态代理类的实例:
/**** 方式一 ****/
//创建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类的静态方法创建的动态代理类的实例具有以下特点:
1. 假定变量foo 是一个动态代理类的实例,并且这个动态代理类实现了Foo 接口,那么“foo instanceof Foo”的值为true。把变量foo强制转换为Foo类型是合法的:
(Foo) foo //合法

2.每个动态代理类实例都和一个InvocationHandler 实例关联。Proxy 类的getInvocationHandler(Object proxy)静态方法返回与参数proxy指定的代理类实例所关联的InvocationHandler 对象。

3.假定Foo接口有一个amethod()方法,那么当程序调用动态代理类实例foo的amethod()方法时,该方法会调用与它关联的InvocationHandler 对象的invoke()方法。

InvocationHandler 接口为方法调用接口,它声明了负责调用任意一个方法的invoke()方法:
Object invoke(Object proxy,Method method,Object[] args) throws Throwable

参数proxy指定动态代理类实例,参数method指定被调用的方法,参数args 指定向被调用方法传递的参数,invoke()方法的返回值表示被调用方法的返回值。

 

最后看一个例子,该例子模仿spring 的AOP原理。

 

 

[java]  view plain copy
  1. package proxy;  
  2.   
  3. import java.lang.reflect.InvocationHandler;  
  4. import java.lang.reflect.Method;  
  5. import java.lang.reflect.Proxy;  
  6.   
  7. class Logic{  
  8.     public void logic(){  
  9.         Dao dao = Factory.create();  
  10.         System.out.println("dynamic proxy's name: "+dao.getClass().getName());  
  11.         dao.insert();  
  12.     }  
  13. }  
  14.   
  15. class Factory{  
  16.     static Dao create(){  
  17.         Dao dao = new JdbcDao();  
  18.         MyInvocationHandler hand = new MyInvocationHandler();  
  19.         return (Dao)hand.get(dao);  
  20.     }  
  21. }  
  22.   
  23. interface Dao{  
  24.     public void update();  
  25.     public void insert();  
  26. }  
  27. class JdbcDao implements Dao{  
  28.     public void update(){  
  29.         System.out.println("in jdbc update");  
  30.     }  
  31.     public void insert(){  
  32.         System.out.println("in jdbc insert");  
  33.     }  
  34. }  
  35. class HibernateDao implements Dao{  
  36.     public void update(){  
  37.         System.out.println("in hibernate update");  
  38.     }  
  39.     public void insert(){  
  40.         System.out.println("in hibernate insert");  
  41.     }  
  42. }  
  43.   
  44. class MyInvocationHandler implements InvocationHandler {  
  45.     Object o;  
  46.     public Object get(Object o){  
  47.         System.out.println("in get method of MyInvocationHandler");  
  48.         this.o = o;  
  49.         return Proxy.newProxyInstance(o.getClass().getClassLoader(),o.getClass().getInterfaces(),this);  
  50.     }  
  51.     public Object invoke(Object arg0, Method arg1, Object[] arg2)  
  52.             throws Throwable {  
  53.         System.out.println("write log before invoke");  
  54.         Object result = arg1.invoke(o, arg2);  
  55.         System.out.println("write log after invoke");  
  56.         return result;  
  57.     }  
  58. }  
  59.   
  60. public class Test {  
  61.     public static void main(String[] args) {  
  62.         Logic l = new Logic();  
  63.         l.logic();  
  64.     }  
  65. }  

运行结果:

 

in get method of MyInvocationHandler
dynamic proxy's name: proxy.$Proxy0
write log before invoke
in jdbc insert
write log after invoke

 

结论: JDK的动态代理并不能随心所欲的代理所有的类。Proxy.newProxyInstance方法的第二个参数只能是接口数组, 也就是Proxy只能代理接口。

原文:http://blog.csdn.net/xuchunlin0216/article/details/3839837

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值