自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

探討Java代理模式與反射機(jī)制的實(shí)際應(yīng)用

開(kāi)發(fā) 后端
Java的代理模式和反射機(jī)制,相信Java的開(kāi)發(fā)者們都非常熟悉,文章會(huì)通過(guò)一個(gè)實(shí)例,深入探討代理模式與Java反射機(jī)制的實(shí)際應(yīng)用。

Java提供了一套機(jī)制來(lái)動(dòng)態(tài)執(zhí)行方法和構(gòu)造方法,以及數(shù)組操作等,這套機(jī)制就叫反射。而代理模式是為其他對(duì)象提供一種代理以控制對(duì)這個(gè)對(duì)象的訪問(wèn),讓我們的目標(biāo)類和代理類實(shí)現(xiàn)同一接口,在代理類中調(diào)用目標(biāo)類對(duì)象的方法。

反射機(jī)制是如今很多Java流行框架的實(shí)現(xiàn)基礎(chǔ),其中包括SpringHibernate等。如果我們將反射機(jī)制加入到Java的代理模式中,就可以實(shí)現(xiàn)一個(gè)公共的代理類,省去我們不少功夫。

  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;    //對(duì)象所屬的類     
  11.     private Object target;  //目標(biāo)對(duì)象     
  12.     private Method method;  //目標(biāo)方法     
  13.     private Object[] params;    //參數(shù)數(shù)組     
  14.          
  15.     @SuppressWarnings("unchecked")     
  16.     public MethodProxy(Object target, String methodName, Object ... params) {     
  17.         rebindTarget(target, methodName, params);   //設(shè)置目標(biāo)對(duì)象與方法     
  18.     }     
  19.          
  20.     /**    
  21.      * 重新設(shè)置目標(biāo)對(duì)象與方法    
  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);   //設(shè)置目標(biāo)方法     
  30.     }     
  31.          
  32.     /**    
  33.      * 重新設(shè)置目標(biāo)方法    
  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.      * 動(dòng)態(tài)調(diào)用已綁定的方法    
  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; //對(duì)象所屬的類  
  80.  private Object target; //目標(biāo)對(duì)象  
  81.  private Method method; //目標(biāo)方法  
  82.  private Object[] params; //參數(shù)數(shù)組  
  83.    
  84.  @SuppressWarnings("unchecked")  
  85.  public MethodProxy(Object target, String methodName, Object ... params) {  
  86.   rebindTarget(target, methodName, params); //設(shè)置目標(biāo)對(duì)象與方法  
  87.  }  
  88.    
  89.  /**  
  90.   * 重新設(shè)置目標(biāo)對(duì)象與方法  
  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); //設(shè)置目標(biāo)方法  
  99.  }  
  100.    
  101.  /**  
  102.   * 重新設(shè)置目標(biāo)方法  
  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.   * 動(dòng)態(tài)調(diào)用已綁定的方法  
  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. 這樣就可以實(shí)現(xiàn)動(dòng)態(tài)地調(diào)用某個(gè)對(duì)象的某個(gè)方法了,寫(xiě)個(gè)測(cè)試代碼如下:  
  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.    

我們通過(guò)代理類來(lái)調(diào)用Manager類中的say()和love()方法,測(cè)試代碼如下:

  1. Manager man = new Manager();    //目標(biāo)對(duì)象     
  2. MethodProxy proxy = new MethodProxy(man, "say");    //方法代理對(duì)象     
  3. proxy.doMethod();   //調(diào)用被代理的方法     
  4. proxy.rebindMethod("love", "Tom", "Marry"); //重新綁定方法     
  5. proxy.doMethod();   //調(diào)用被代理的方法   
  6.  
  7. Manager man = new Manager(); //目標(biāo)對(duì)象  
  8. MethodProxy proxy = new MethodProxy(man, "say"); //方法代理對(duì)象  
  9. proxy.doMethod(); //調(diào)用被代理的方法  
  10. proxy.rebindMethod("love", "Tom", "Marry"); //重新綁定方法  
  11. proxy.doMethod(); //調(diào)用被代理的方法 

#p#

這樣就實(shí)現(xiàn)了動(dòng)態(tài)代理調(diào)用對(duì)象的方法,上面代碼輸出結(jié)果就不貼出來(lái)了。如果要設(shè)置前置通知和后置通知等功能,也很容易實(shí)現(xiàn),只需在“proxy.doMethod()”代碼處的前面和后面設(shè)置即行。擴(kuò)展應(yīng)用:我們?cè)谏厦娴腗ethodProxy類中加入以下方法:

  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.  } 

這個(gè)方法用來(lái)讀取方法上的注解(Annotation),有什么用呢?我們寫(xiě)一個(gè)注解來(lái)測(cè)試下。 

  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();   

我們要引進(jìn)Annotation相關(guān)的類: 

  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; 

我們另外寫(xiě)一個(gè)測(cè)試用的業(yè)務(wù)類: 

  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. }  

接寫(xiě)上例中的proxy對(duì)象測(cè)試代碼: 

  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); //重新綁定對(duì)象和方法     
  5. Low low = (Low)proxy.getAnnotation(Low.class);     
  6. if(boy.getAge() < low.boyAge()) {     
  7.     System.out.println(boy.getName() + "還不到法定年齡,不能談戀愛(ài)!");     
  8. } else if(girl.getAge() < low.girlAge()) {     
  9.     System.out.println(girl.getName() + "還不到法定年齡,不能談戀愛(ài)!");     
  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); //重新綁定對(duì)象和方法  
  18.   Low low = (Low)proxy.getAnnotation(Low.class);  
  19.   if(boy.getAge() < low.boyAge()) {  
  20.    System.out.println(boy.getName() + "還不到法定年齡,不能談戀愛(ài)!");  
  21.   } else if(girl.getAge() < low.girlAge()) {  
  22.    System.out.println(girl.getName() + "還不到法定年齡,不能談戀愛(ài)!");  
  23.   } else {  
  24.    proxy.doMethod();  
  25.   } 

這就實(shí)現(xiàn)了,通過(guò)Java的反射機(jī)制來(lái)讀取Annotation的值,并根據(jù)Annotation的值,來(lái)處理業(yè)務(wù)數(shù)據(jù)有效性的判斷,或者面向切面動(dòng)態(tài)地注入對(duì)象,或者作日志、攔截器等等。這種用法在所多框架中都常常看到, 我們?cè)陂_(kāi)發(fā)自己的Java組件時(shí),不妨也采用一下吧!

【編輯推薦】

  1. Java設(shè)計(jì)模式之虛擬代理模式
  2. 關(guān)于Java反射機(jī)制的一個(gè)實(shí)例
  3. Java實(shí)例講解反射機(jī)制Reflection
  4. JAVA反射機(jī)制的簡(jiǎn)單應(yīng)用
責(zé)任編輯:王曉東 來(lái)源: javaeye
相關(guān)推薦

2017-03-24 09:44:33

Java反射機(jī)制

2017-05-17 15:28:15

Java反射機(jī)制

2012-02-08 10:12:19

Java反射

2024-01-04 07:42:44

JavaCGLIBJDK

2011-05-23 09:59:04

網(wǎng)絡(luò)綜合布線系統(tǒng)布線

2009-12-15 09:34:09

路由信息協(xié)議

2011-11-24 21:03:10

ibmdw

2009-11-06 14:26:42

無(wú)線局域網(wǎng)

2012-02-08 10:37:42

Java反射

2010-02-04 11:15:28

C++模板限制

2009-08-13 16:57:37

.NET緩存機(jī)制

2021-09-06 10:04:47

觀察者模式應(yīng)用

2012-04-05 13:50:38

Java

2023-10-17 09:26:44

Java工具

2011-04-01 14:50:56

Java的反射機(jī)制

2014-07-30 09:48:23

OpenflowSDN

2011-07-04 16:48:56

JAVA垃圾回收機(jī)制GC

2009-10-20 14:51:57

網(wǎng)絡(luò)綜合布線系統(tǒng)

2010-03-18 13:49:40

ATM交換機(jī)

2009-12-10 15:50:12

IP多播路由協(xié)議
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)