成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java 動態代理(Proxy)

開發 后端
主要是通過 java.lang.reflect.Proxy 類和 java.lang.reflect.InvocationHandler 接口。 Proxy 類主要用來獲取動態代理對象,InvocationHandler 接口用來約束調用者行為。

動態代理可以提供對另一個對象的訪問,同時隱藏實際對象的具體事實,代理對象對客戶隱藏了實際對象。動態代理可以對請求進行其他的一些處理,在不允許直接訪問某些類,或需要對訪問做一些特殊處理等,這時候可以考慮使用代理。目前 Java 開發包中提供了對動態代理的支持,但現在只支持對接口的實現。

主要是通過 java.lang.reflect.Proxy 類和 java.lang.reflect.InvocationHandler 接口。 Proxy 類主要用來獲取動態代理對象,InvocationHandler 接口用來約束調用者行為。

“寫一個 ArrayList 類的代理,其內部實現和 ArrayList 中完全相同的功能,并可以計算每個方法運行的時間。”這是一份考題上的題目,沒有答案,來看下實現:

  1. package example;  
  2. import java.lang.reflect.InvocationHandler;  
  3. import java.lang.reflect.Method;  
  4. import java.lang.reflect.Proxy;  
  5. import java.util.ArrayList;  
  6. import java.util.List;  
  7. import java.util.concurrent.TimeUnit;  
  8. /**  
  9.  * -----------------------------------------  
  10.  * @描述  TODO  
  11.  * @作者  fancy  
  12.  * @郵箱  fancydeepin@yeah.net  
  13.  * @日期  2012-8-27 <p>  
  14.  * -----------------------------------------  
  15.  */ 
  16. public class ProxyApp {  
  17.  
  18.  
  19.     public static void main(String[] args){  
  20.           
  21.         //ArrayList代理,通過代理計算每個方法調用所需時間  
  22.         List<Integer> arrayListProxy = (List<Integer>)Proxy.newProxyInstance(  
  23.             ArrayList.class.getClassLoader(),   /*定義代理類的類加載器,用于創建代理對象,不一定必須是ArrayList,也可以是其他的類加載器*/ 
  24.             ArrayList.class.getInterfaces(),     /*代理類要實現的接口列表*/ 
  25.             new InvocationHandler() {            /*指派方法調用的調用處理程序,這里用了匿名內部類*/ 
  26.                   
  27.                 private ArrayList<Integer> target = new ArrayList<Integer>(); //目標對象(真正操作的對象)  
  28.                 /**  
  29.                  * <B>方法描述:</B>  
  30.                  * <p style="margin-left:20px;color:#A52A2A;">  
  31.                  * 在代理實例上處理方法調用并返回結果  
  32.                  * @param proxy     代理對象(注意不是目標對象)  
  33.                  * @param method  被代理的方法  
  34.                  * @param args         被代理的方法的參數集  
  35.                  * @return <span style="color: #008080;"> 返回方法調用結果 </span>  
  36.                  */ 
  37.                 @Override 
  38.                 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
  39.                       
  40.                     long beginTime = System.currentTimeMillis();  //開始時間  
  41.                     TimeUnit.MICROSECONDS.sleep(1);  
  42.                     Object obj = method.invoke(target, args);          //實際調用的方法,并接受方法的返回值  
  43.                     long endTime = System.currentTimeMillis();   //結束時間  
  44.                     System.out.println("[" + method.getName() + "] spend " + (endTime - beginTime) + " ms");  
  45.                     return obj;   //返回實際調用的方法的返回值  
  46.                       
  47.                 }  
  48.                   
  49.             }  
  50.         );  
  51.         arrayListProxy.add(2);  
  52.         arrayListProxy.add(4);  
  53.         System.out.println("--------- 迭代 ---------");  
  54.         for(int i : arrayListProxy){  
  55.             System.out.print(i + "\t");  
  56.         }  
  57.     }  

后臺打印輸出結果:

 [add] spend 2 ms
[add] spend 1 ms
--------- 迭代 ---------
[iterator] spend 1 ms
2 4

從代碼上來看,用到了匿名內部類,這樣一來,InvocationHandler 只能用一次,如果多個地方都需要用到這樣一個相同的 InvocationHandler,可以將其抽象出來成為一個單獨的類:

  1. package test;  
  2. import java.lang.reflect.InvocationHandler;  
  3. import java.lang.reflect.Method;  
  4. import java.util.concurrent.TimeUnit;  
  5. public class MyInvocationHandler implements InvocationHandler{  
  6.     private Object target; //目標對象      
  7.     public MyInvocationHandler(Object target){  
  8.         this.target = target;  
  9.     }  
  10.     @Override 
  11.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {          
  12.         long beginTime = System.currentTimeMillis();  
  13.         TimeUnit.MICROSECONDS.sleep(1);  
  14.         Object obj = method.invoke(target, args);  
  15.         long endTime = System.currentTimeMillis();  
  16.         System.out.println("[" + method.getName() + "] spend " + (endTime - beginTime) + " ms");  
  17.         return obj;      
  18.     }  

客戶端調用改成:

  1. package example;  
  2. import java.lang.reflect.Proxy;  
  3. import java.util.ArrayList;  
  4. import java.util.List;  
  5. /**  
  6.  * -----------------------------------------  
  7.  * @描述  TODO  
  8.  * @作者  fancy  
  9.  * @郵箱  fancydeepin@yeah.net  
  10.  * @日期  2012-8-27 <p>  
  11.  * -----------------------------------------  
  12.  */ 
  13. public class ProxyApp {  
  14.     public static void main(String[] args){          
  15.         //ArrayList代理,通過代理計算每個方法調用所需時間  
  16.         List<Integer> arrayListProxy = (List<Integer>)Proxy.newProxyInstance(  
  17.             ArrayList.class.getClassLoader(),     /*定義代理類的類加載器,用于創建代理對象,不一定必須是ArrayList,也可以是其他的類加載器*/ 
  18.             ArrayList.class.getInterfaces(),       /*代理類要實現的接口列表*/ 
  19.             new MyInvocationHandler(new ArrayList<Integer>())         /*指派方法調用的調用處理程序,這里用了匿名內部類*/ 
  20.         );  
  21.         arrayListProxy.add(2);  
  22.         arrayListProxy.add(4);  
  23.         System.out.println("--------- 迭代 ---------");  
  24.         for(int i : arrayListProxy){  
  25.             System.out.print(i + "\t");  
  26.         }  
  27.     }  

從上面代碼看來,客戶端知道代理的實際目標對象,還知道怎么樣去創建這樣一個代理對象,如果想把這些信息全部對客戶端隱藏起來,可以將這些代碼挪到一個類中,將它們封裝起來:

  1. package example;  
  2. import java.lang.reflect.InvocationHandler;  
  3. import java.lang.reflect.Method;  
  4. import java.lang.reflect.Proxy;  
  5. import java.util.ArrayList;  
  6. import java.util.List;  
  7. import java.util.concurrent.TimeUnit;  
  8. /**  
  9.  * -----------------------------------------  
  10.  * @描述  TODO  
  11.  * @作者  fancy  
  12.  * @郵箱  fancydeepin@yeah.net  
  13.  * @日期  2012-8-27 <p>  
  14.  * -----------------------------------------  
  15.  */ 
  16. public class ProxyUtil {  
  17.     public enum ArrayListProxy {  
  18.         PROXY;       
  19.         private Object target;    
  20.         ArrayListProxy(){  
  21.             this.target = new ArrayList<Object>();  
  22.         }  
  23.         public List getInstance(){  
  24.               
  25.             return (List)Proxy.newProxyInstance(ArrayList.class.getClassLoader(), ArrayList.class.getInterfaces(),  
  26.                     new InvocationHandler() {     
  27.                         @Override 
  28.                         public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
  29.                               
  30.                             long beginTime = System.currentTimeMillis();  
  31.                             TimeUnit.MICROSECONDS.sleep(1);  
  32.                             Object obj = method.invoke(target, args);  
  33.                             long endTime = System.currentTimeMillis();  
  34.                             System.out.println("[" + method.getName() + "] spend " + (endTime - beginTime) + " ms");  
  35.                             return obj;   
  36.                         }  
  37.                     });  
  38.         }  
  39.     }  

客戶端調用改成:

  1. package example;  
  2. import java.util.List;  
  3. import example.ProxyUtil.ArrayListProxy;  
  4. /**  
  5.  * -----------------------------------------  
  6.  * @描述  TODO  
  7.  * @作者  fancy  
  8.  * @郵箱  fancydeepin@yeah.net  
  9.  * @日期  2012-8-27 <p>  
  10.  * -----------------------------------------  
  11.  */ 
  12. public class ProxyApp {  
  13.     public static void main(String[] args){  
  14.         List<Integer> arrayListProxy = ArrayListProxy.PROXY.getInstance();  
  15.         arrayListProxy.add(2);  
  16.         arrayListProxy.add(4);  
  17.         System.out.println("--------- 迭代 ---------");  
  18.         for(int i : arrayListProxy){  
  19.             System.out.print(i + "\t");  
  20.         }  
  21.     }  

上面代碼中用到了枚舉 enum,如果不想用枚舉,就改用普通類來實現就行了。

原文鏈接:http://www.blogjava.net/fancydeepin/archive/2012/08/27/java_proxy.html

【編輯推薦】

  1. 近期面試經歷總結及公司評價(上)
  2. 函數式編程是一個倒退
  3. 為什么Java程序占用的內存比實際分配的多
  4. Java 8 Lambda:模擬Mixin實現類多重繼承
  5. Java項目經驗——程序員成長的關鍵
責任編輯:張偉 來源: fancydeepin的博客
相關推薦

2011-04-06 11:41:25

Java動態代理

2017-10-12 14:56:11

2011-03-23 10:40:51

java代理模式

2023-12-06 08:23:44

代理模式設計模式

2015-09-22 11:09:47

Java 8動態代理

2021-07-06 06:39:22

Java靜態代理動態代理

2017-05-11 21:30:01

Android動態代理ServiceHook

2023-02-24 07:42:30

Java動態代理

2015-09-28 15:59:00

Java動態代理機制

2011-11-17 14:32:45

Java靜態代理動態代理

2010-12-02 13:36:48

Paros proxy網頁程序漏洞

2015-09-24 08:55:14

Java動態代理擴展

2015-09-24 08:54:36

java動態代理

2010-04-21 09:26:54

Java動態代理

2020-04-27 07:28:00

Java反射動態代理

2012-02-08 10:12:19

Java反射

2020-12-28 07:47:35

動態代理AOP

2022-01-26 00:05:00

AOPRPC遠程調用

2009-06-22 15:10:00

java 編程AOP

2012-08-10 13:55:56

Java動態代理
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久精品视频在线播放 | 超碰最新在线 | 国产精品毛片一区二区三区 | 污片在线免费观看 | 国产性生活一级片 | 中文字幕一区在线观看视频 | 视频一区二区在线观看 | 欧美日韩亚| 99视频在线免费观看 | 麻豆精品一区二区三区在线观看 | 国产精品乱码一区二区三区 | 欧美 日韩 亚洲91麻豆精品 | 亚洲国产高清高潮精品美女 | 在线观看中文字幕dvd播放 | 拍戏被cao翻了h承欢 | 免费在线h视频 | 久草免费在线视频 | 婷婷桃色网 | 国产高清在线精品一区二区三区 | 亚洲成av人片在线观看 | 欧美精品中文字幕久久二区 | 黄网站免费在线观看 | 一区二区三区在线观看视频 | 免费视频一区二区 | 黄色精品 | 亚洲乱码一区二区三区在线观看 | 日韩精品中文字幕在线 | 中文字幕亚洲精品 | 日韩av一区二区在线 | 欧美一区二区三区视频在线观看 | 国产在线观看一区二区三区 | 日韩亚洲欧美综合 | 天天艹逼网 | 羞羞的视频免费在线观看 | 亚洲视频一区在线播放 | 99综合| 九九热免费看 | 久久久久国产精品 | 在线久草| 亚洲精品大片 | 精品日韩一区二区三区 |