动态代理模式 动态代理模式

2.动态代理类

        Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类:

        (1). Interface InvocationHandler:该接口中仅定义了一个方法Objectinvoke(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接口,以第一节中的示例为例:

 

扫描二维码关注公众号,回复: 731101 查看本文章

      通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject)也可以动态改变,从而实现了非常灵活的动态代理关系。

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功能:代理模式中公共的行为接口 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public interface Subject {  
  10.     public abstract void request();  
  11. }  

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功能:代理模式中真正的实现 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public class RealSubject implements Subject {     
  10.     /** 
  11.      * 构造函数 
  12.      */  
  13.     public RealSubject(){     
  14.     }  
  15.     /* (non-Javadoc) 
  16.      * @see six.proxytrip.Subject#request() 
  17.      */  
  18.     public void request() {  
  19.         System.out.println("this is a real request!");    
  20.     }  
  21. }  

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功能:代理类 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public class ProxySubject implements InvocationHandler {  
  10.     /** 
  11.      * 声明对象 
  12.      */  
  13.     private Object subject;  
  14.     /** 
  15.      *  
  16.      */  
  17.     public ProxySubject() {  
  18.     }  
  19.     public ProxySubject(Object obj) {  
  20.         this.subject = obj;  
  21.     }  
  22.     public Object invoke(Object proxy, Method method, Object[] args)  
  23.             throws Throwable {  
  24.         System.out.println( " before calling  "   +  method);  
  25.         method.invoke(subject,args);  
  26.         System.out.println( " after calling  "   +  method);  
  27.         return   null ;  
  28.     }  
  29. }  

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功 能 : 客户端测试代码 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public class Client {  
  10.     /** 
  11.      * @param args 
  12.      */  
  13.     public static void main(String[] args) {  
  14.          RealSubject rs  =   new  RealSubject();  // 在这里指定被代理类    
  15.            InvocationHandler ds  =   new  ProxySubject(rs);   
  16.            Class cls  =  rs.getClass();   
  17.             // 以下是一次性生成代理   
  18.            Subject subject  =  (Subject) Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(),ds );   
  19.            subject.request();   
  20.   
  21.     }  
  22. }  

 

Java代码    收藏代码
  1. 测试结果:  
  2.  before calling  public abstract void six.dydmicproxy.Subject.request()  
  3. this is a real request!  
  4.  after calling  public abstract void six.dydmicproxy.Subject.request()  

     该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;此外,在该类还实现了invoke方法,该方法中的

method.invoke(sub,args);

其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,args为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。

2.动态代理类

        Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类:

        (1). Interface InvocationHandler:该接口中仅定义了一个方法Objectinvoke(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)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject)也可以动态改变,从而实现了非常灵活的动态代理关系。

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功能:代理模式中公共的行为接口 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public interface Subject {  
  10.     public abstract void request();  
  11. }  

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功能:代理模式中真正的实现 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public class RealSubject implements Subject {     
  10.     /** 
  11.      * 构造函数 
  12.      */  
  13.     public RealSubject(){     
  14.     }  
  15.     /* (non-Javadoc) 
  16.      * @see six.proxytrip.Subject#request() 
  17.      */  
  18.     public void request() {  
  19.         System.out.println("this is a real request!");    
  20.     }  
  21. }  

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功能:代理类 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public class ProxySubject implements InvocationHandler {  
  10.     /** 
  11.      * 声明对象 
  12.      */  
  13.     private Object subject;  
  14.     /** 
  15.      *  
  16.      */  
  17.     public ProxySubject() {  
  18.     }  
  19.     public ProxySubject(Object obj) {  
  20.         this.subject = obj;  
  21.     }  
  22.     public Object invoke(Object proxy, Method method, Object[] args)  
  23.             throws Throwable {  
  24.         System.out.println( " before calling  "   +  method);  
  25.         method.invoke(subject,args);  
  26.         System.out.println( " after calling  "   +  method);  
  27.         return   null ;  
  28.     }  
  29. }  

 

Java代码    收藏代码
  1. /** 
  2.  *  
  3.  * @功 能 : 客户端测试代码 
  4.  * @创建人 gao_jie 
  5.  * @创建日期 May 8, 2009 
  6.  * @版本 1.0 
  7.  *  
  8.  */  
  9. public class Client {  
  10.     /** 
  11.      * @param args 
  12.      */  
  13.     public static void main(String[] args) {  
  14.          RealSubject rs  =   new  RealSubject();  // 在这里指定被代理类    
  15.            InvocationHandler ds  =   new  ProxySubject(rs);   
  16.            Class cls  =  rs.getClass();   
  17.             // 以下是一次性生成代理   
  18.            Subject subject  =  (Subject) Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(),ds );   
  19.            subject.request();   
  20.   
  21.     }  
  22. }  

 

Java代码    收藏代码
  1. 测试结果:  
  2.  before calling  public abstract void six.dydmicproxy.Subject.request()  
  3. this is a real request!  
  4.  after calling  public abstract void six.dydmicproxy.Subject.request()  

     该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;此外,在该类还实现了invoke方法,该方法中的

method.invoke(sub,args);

其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,args为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。

猜你喜欢

转载自rongdmmap-126-com.iteye.com/blog/1433273
今日推荐