探讨Java代理模式与反射机制的实际应用

开发 后端
Java的代理模式和反射机制,相信Java的开发者们都非常熟悉,文章会通过一个实例,深入探讨代理模式与Java反射机制的实际应用。

Java提供了一套机制来动态执行方法和构造方法,以及数组操作等,这套机制就叫反射。而代理模式是为其他对象提供一种代理以控制对这个对象的访问,让我们的目标类和代理类实现同一接口,在代理类中调用目标类对象的方法。

反射机制是如今很多Java流行框架的实现基础,其中包括SpringHibernate等。如果我们将反射机制加入到Java的代理模式中,就可以实现一个公共的代理类,省去我们不少功夫。

  1. import java.lang.reflect.InvocationTargetException;     
  2. import java.lang.reflect.Method;     
  3. /**    
  4.  * 方法代理类    
  5.  * @author rongxinhua    
  6.  *    
  7.  */    
  8. public class MethodProxy {     
  9.          
  10.     private Class clazz;    //对象所属的类     
  11.     private Object target;  //目标对象     
  12.     private Method method;  //目标方法     
  13.     private Object[] params;    //参数数组     
  14.          
  15.     @SuppressWarnings("unchecked")     
  16.     public MethodProxy(Object target, String methodName, Object ... params) {     
  17.         rebindTarget(target, methodName, params);   //设置目标对象与方法     
  18.     }     
  19.          
  20.     /**    
  21.      * 重新设置目标对象与方法    
  22.      * @param target    
  23.      * @param methodName    
  24.      * @param params    
  25.      */    
  26.     public void rebindTarget(Object target, String methodName, Object ... params) {     
  27.         this.target = target;     
  28.         this.clazz = target.getClass();     
  29.         rebindMethod(methodName, params);   //设置目标方法     
  30.     }     
  31.          
  32.     /**    
  33.      * 重新设置目标方法    
  34.      * @param methodName    
  35.      * @param params    
  36.      */    
  37.     public void rebindMethod(String methodName, Object ...params) {     
  38.         this.params = params;     
  39.         int paramLength = params.length;     
  40.         Class[] paramTypes = new Class[paramLength];     
  41.         for(int i = 0 ; i < paramLength ; i ++ ) {     
  42.             paramTypes[i] = params[i].getClass();     
  43.         }     
  44.         try {     
  45.             this.method = clazz.getMethod(methodName, paramTypes);     
  46.         } catch (SecurityException e) {     
  47.             e.printStackTrace();     
  48.         } catch (NoSuchMethodException e) {     
  49.             e.printStackTrace();     
  50.         }     
  51.     }     
  52.          
  53.     /**    
  54.      * 动态调用已绑定的方法    
  55.      */    
  56.     public void doMethod() {     
  57.         try {     
  58.             this.method.invoke(target, params);     
  59.         } catch (IllegalArgumentException e) {     
  60.             e.printStackTrace();     
  61.         } catch (IllegalAccessException e) {     
  62.             e.printStackTrace();     
  63.         } catch (InvocationTargetException e) {     
  64.             e.printStackTrace();     
  65.         }     
  66.     }     
  67.     
  68. }   
  69.  
  70. import java.lang.reflect.InvocationTargetException;  
  71. import java.lang.reflect.Method;  
  72. /**  
  73.  * 方法代理类  
  74.  * @author rongxinhua  
  75.  *  
  76.  */  
  77. public class MethodProxy {  
  78.    
  79.  private Class clazz; //对象所属的类  
  80.  private Object target; //目标对象  
  81.  private Method method; //目标方法  
  82.  private Object[] params; //参数数组  
  83.    
  84.  @SuppressWarnings("unchecked")  
  85.  public MethodProxy(Object target, String methodName, Object ... params) {  
  86.   rebindTarget(target, methodName, params); //设置目标对象与方法  
  87.  }  
  88.    
  89.  /**  
  90.   * 重新设置目标对象与方法  
  91.   * @param target  
  92.   * @param methodName  
  93.   * @param params  
  94.   */  
  95.  public void rebindTarget(Object target, String methodName, Object ... params) {  
  96.   this.target = target;  
  97.   this.clazz = target.getClass();  
  98.   rebindMethod(methodName, params); //设置目标方法  
  99.  }  
  100.    
  101.  /**  
  102.   * 重新设置目标方法  
  103.   * @param methodName  
  104.   * @param params  
  105.   */  
  106.  public void rebindMethod(String methodName, Object ...params) {  
  107.   this.params = params;  
  108.   int paramLength = params.length;  
  109.   Class[] paramTypes = new Class[paramLength];  
  110.   for(int i = 0 ; i < paramLength ; i ++ ) {  
  111.    paramTypes[i] = params[i].getClass();  
  112.   }  
  113.   try {  
  114.    this.method = clazz.getMethod(methodName, paramTypes);  
  115.   } catch (SecurityException e) {  
  116.    e.printStackTrace();  
  117.   } catch (NoSuchMethodException e) {  
  118.    e.printStackTrace();  
  119.   }  
  120.  }  
  121.    
  122.  /**  
  123.   * 动态调用已绑定的方法  
  124.   */  
  125.  public void doMethod() {  
  126.   try {  
  127.    this.method.invoke(target, params);  
  128.   } catch (IllegalArgumentException e) {  
  129.    e.printStackTrace();  
  130.   } catch (IllegalAccessException e) {  
  131.    e.printStackTrace();  
  132.   } catch (InvocationTargetException e) {  
  133.    e.printStackTrace();  
  134.   }  
  135.  }  
  136.  
  137. }  
  138.  
  139. 这样就可以实现动态地调用某个对象的某个方法了,写个测试代码如下:  
  140.  
  141. public class Manager {     
  142.          
  143.     public void say() {     
  144.         System.out.println("Nobody say nothing");     
  145.     }     
  146.          
  147.     public void love(String boy, String girl) {     
  148.         System.out.println(boy + " love " + girl);     
  149.     }     
  150.          
  151. }   
  152.  
  153. public class Manager {  
  154.    
  155.  public void say() {  
  156.   System.out.println("Nobody say nothing");  
  157.  }  
  158.    
  159.  public void love(String boy, String girl) {  
  160.   System.out.println(boy + " love " + girl);  
  161.  }  
  162.    

我们通过代理类来调用Manager类中的say()和love()方法,测试代码如下:

  1. Manager man = new Manager();    //目标对象     
  2. MethodProxy proxy = new MethodProxy(man, "say");    //方法代理对象     
  3. proxy.doMethod();   //调用被代理的方法     
  4. proxy.rebindMethod("love", "Tom", "Marry"); //重新绑定方法     
  5. proxy.doMethod();   //调用被代理的方法   
  6.  
  7. Manager man = new Manager(); //目标对象  
  8. MethodProxy proxy = new MethodProxy(man, "say"); //方法代理对象  
  9. proxy.doMethod(); //调用被代理的方法  
  10. proxy.rebindMethod("love", "Tom", "Marry"); //重新绑定方法  
  11. proxy.doMethod(); //调用被代理的方法 

#p#

这样就实现了动态代理调用对象的方法,上面代码输出结果就不贴出来了。如果要设置前置通知和后置通知等功能,也很容易实现,只需在“proxy.doMethod()”代码处的前面和后面设置即行。扩展应用:我们在上面的MethodProxy类中加入以下方法:

  1. /**    
  2.  * 获取方法上的注解    
  3.  * @param anClazz 注解类    
  4.  * @return    
  5.  */    
  6. public Annotation getAnnotation(Class anClazz) {     
  7.     return this.method.getAnnotation(anClazz);     
  8. }   
  9.  
  10.  /**  
  11.   * 获取方法上的注解  
  12.   * @param anClazz 注解类  
  13.   * @return  
  14.   */  
  15.  public Annotation getAnnotation(Class anClazz) {  
  16.   return this.method.getAnnotation(anClazz);  
  17.  } 

这个方法用来读取方法上的注解(Annotation),有什么用呢?我们写一个注解来测试下。 

  1. @Retention(RetentionPolicy.RUNTIME)     
  2. @Target(ElementType.METHOD)     
  3. @interface Low {     
  4.     int boyAge();      
  5.     int girlAge();       
  6. }   
  7.  
  8. @Retention(RetentionPolicy.RUNTIME)  
  9. @Target(ElementType.METHOD)  
  10. @interface Low {  
  11.  int boyAge();   
  12.  int girlAge();   

我们要引进Annotation相关的类: 

  1. import java.lang.annotation.Annotation;     
  2. import java.lang.annotation.ElementType;     
  3. import java.lang.annotation.Retention;     
  4. import java.lang.annotation.RetentionPolicy;     
  5. import java.lang.annotation.Target;   
  6.  
  7. import java.lang.annotation.Annotation;  
  8. import java.lang.annotation.ElementType;  
  9. import java.lang.annotation.Retention;  
  10. import java.lang.annotation.RetentionPolicy;  
  11. import java.lang.annotation.Target; 

我们另外写一个测试用的业务类: 

  1. public class LoveManager {     
  2.          
  3.     @Low(boyAge=12girlAge=10)     
  4.     public void beAbleToLove(Person boy, Person girl) {     
  5.         System.out.println(boy.getName() + " is able to love " + girl.getName());     
  6.     }     
  7.          
  8. }     
  9.     
  10. public class Person {     
  11.     private String name;     
  12.     private int age;     
  13.     public Person(String name, int age) {     
  14.         this.name = name;     
  15.         this.age = age;     
  16.     }     
  17.     //getter方法略     
  18. }   
  19.  
  20. public class LoveManager {  
  21.    
  22.  @Low(boyAge=12girlAge=10)  
  23.  public void beAbleToLove(Person boy, Person girl) {  
  24.   System.out.println(boy.getName() + " is able to love " + girl.getName());  
  25.  }  
  26.    
  27. }  
  28.  
  29. public class Person {  
  30.  private String name;  
  31.  private int age;  
  32.  public Person(String name, int age) {  
  33.   this.name = name;  
  34.   this.age = age;  
  35.  }  
  36.  //getter方法略  
  37. }  

接写上例中的proxy对象测试代码: 

  1. LoveManager loveManager = new LoveManager();     
  2. Person boy = new Person("Tom", 13);     
  3. Person girl = new Person("Marry", 10);     
  4. proxy.rebindTarget(loveManager, "beAbleToLove", boy, girl); //重新绑定对象和方法     
  5. Low low = (Low)proxy.getAnnotation(Low.class);     
  6. if(boy.getAge() < low.boyAge()) {     
  7.     System.out.println(boy.getName() + "还不到法定年龄,不能谈恋爱!");     
  8. } else if(girl.getAge() < low.girlAge()) {     
  9.     System.out.println(girl.getName() + "还不到法定年龄,不能谈恋爱!");     
  10. } else {     
  11.     proxy.doMethod();     
  12. }   
  13.  
  14.   LoveManager loveManager = new LoveManager();  
  15.   Person boy = new Person("Tom", 13);  
  16.   Person girl = new Person("Marry", 10);  
  17.   proxy.rebindTarget(loveManager, "beAbleToLove", boy, girl); //重新绑定对象和方法  
  18.   Low low = (Low)proxy.getAnnotation(Low.class);  
  19.   if(boy.getAge() < low.boyAge()) {  
  20.    System.out.println(boy.getName() + "还不到法定年龄,不能谈恋爱!");  
  21.   } else if(girl.getAge() < low.girlAge()) {  
  22.    System.out.println(girl.getName() + "还不到法定年龄,不能谈恋爱!");  
  23.   } else {  
  24.    proxy.doMethod();  
  25.   } 

这就实现了,通过Java的反射机制来读取Annotation的值,并根据Annotation的值,来处理业务数据有效性的判断,或者面向切面动态地注入对象,或者作日志、拦截器等等。这种用法在所多框架中都常常看到, 我们在开发自己的Java组件时,不妨也采用一下吧!

【编辑推荐】

  1. Java设计模式之虚拟代理模式
  2. 关于Java反射机制的一个实例
  3. Java实例讲解反射机制Reflection
  4. JAVA反射机制的简单应用
责任编辑:王晓东 来源: javaeye
相关推荐

2017-03-24 09:44:33

Java反射机制

2017-05-17 15:28:15

Java反射机制

2012-02-08 10:12:19

Java反射

2024-01-04 07:42:44

JavaCGLIBJDK

2011-05-23 09:59:04

网络综合布线系统布线

2009-12-15 09:34:09

路由信息协议

2012-02-08 10:37:42

Java反射

2009-11-06 14:26:42

无线局域网

2011-11-24 21:03:10

ibmdw

2010-02-04 11:15:28

C++模板限制

2009-08-13 16:57:37

.NET缓存机制

2012-04-05 13:50:38

Java

2021-09-06 10:04:47

观察者模式应用

2023-10-17 09:26:44

Java工具

2011-04-01 14:50:56

Java的反射机制

2011-07-04 16:48:56

JAVA垃圾回收机制GC

2014-07-30 09:48:23

OpenflowSDN

2009-10-20 14:51:57

网络综合布线系统

2010-03-18 13:49:40

ATM交换机

2009-12-10 15:50:12

IP多播路由协议
点赞
收藏

51CTO技术栈公众号