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

十分鐘理解 Java 中的動態代理

開發 后端
若代理類在程序運行前就已經存在,那么這種代理方式被成為 靜態代理 ,這種情況下的代理類通常都是我們在Java代碼中定義的。 通常情況下, 靜態代理中的代理類和委托類會實現同一接口或是派生自相同的父類。

 

一、概述

1. 什么是代理

我們大家都知道微商代理,簡單地說就是代替廠家賣商品,廠家“委托”代理為其銷售商品。關于微商代理,首先我們從他們那里買東西時通常不知道背后的廠家究竟是誰,也就是說,“委托者”對我們來說是不可見的;其次,微商代理主要以朋友圈的人為目標客戶,這就相當于為廠家做了一次對客戶群體的“過濾”。我們把微商代理和廠家進一步抽象,前者可抽象為代理類,后者可抽象為委托類(被代理類)。通過使用代理,通常有兩個優點,并且能夠分別與我們提到的微商代理的兩個特點對應起來:

優點一:可以隱藏委托類的實現;

優點二:可以實現客戶與委托類間的解耦,在不修改委托類代碼的情況下能夠做一些額外的處理。

2. 靜態代理

若代理類在程序運行前就已經存在,那么這種代理方式被成為 靜態代理 ,這種情況下的代理類通常都是我們在Java代碼中定義的。 通常情況下, 靜態代理中的代理類和委托類會實現同一接口或是派生自相同的父類。 下面我們用Vendor類代表生產廠家,BusinessAgent類代表微商代理,來介紹下靜態代理的簡單實現,委托類和代理類都實現了Sell接口,Sell接口的定義如下:

 

  1. public interface Sell { void sell(); void ad(); 
  2.  

Vendor類的定義如下:

 

  1. public class Vendor implements Sell { public void sell() { 
  2.  
  3. System.out.println("In sell method"); 
  4.  
  5. public void ad() { 
  6.  
  7. System,out.println("ad method"
  8.  
  9.  

代理類BusinessAgent的定義如下:

 

  1. public class Vendor implements Sell { public void sell() { 
  2.  
  3. System.out.println("In sell method"); 
  4.  
  5. public void ad() { 
  6.  
  7. System,out.println("ad method"
  8.  
  9.  

 

從BusinessAgent類的定義我們可以了解到,靜態代理可以通過聚合來實現,讓代理類持有一個委托類的引用即可。

下面我們考慮一下這個需求:給Vendor類增加一個過濾功能,只賣貨給大學生。通過靜態代理,我們無需修改Vendor類的代碼就可以實現,只需在BusinessAgent類中的sell方法中添加一個判斷即可如下所示:

 

  1. public class BusinessAgent implements Sell { 
  2.  
  3. ... 
  4.  
  5. public void sell() { 
  6.  
  7. if (isCollegeStudent()) { 
  8.  
  9. vendor.sell(); 
  10.  
  11.  
  12.  
  13. ... 
  14.  

這對應著我們上面提到的使用代理的第二個優點:可以實現客戶與委托類間的解耦,在不修改委托類代碼的情況下能夠做一些額外的處理。靜態代理的局限在于運行前必須編寫好代理類,下面我們重點來介紹下運行時生成代理類的動態代理方式。

二、動態代理

1. 什么是動態代理

代理類在程序運行時創建的代理方式被成為 動態代理。 也就是說,這種情況下,代理類并不是在Java代碼中定義的,而是在運行時根據我們在Java代碼中的“指示”動態生成的。相比于靜態代理, 動態代理的優勢在于可以很方便的對代理類的函數進行統一的處理,而不用修改每個代理類的函數。 這么說比較抽象,下面我們結合一個實例來介紹一下動態代理的這個優勢是怎么體現的。

現在,假設我們要實現這樣一個需求:在執行委托類中的方法之前輸出“before”,在執行完畢后輸出“after”。我們還是以上面例子中的Vendor類作為委托類,BusinessAgent類作為代理類來進行介紹。首先我們來使用靜態代理來實現這一需求,相關代碼如下:

 

  1. public class BusinessAgent implements Sell { 
  2.  
  3. private Vendor mVendor; 
  4.  
  5. public BusinessAgent(Vendor vendor) { 
  6.  
  7. this.mVendor = vendor; 
  8.  
  9.  
  10. public void sell() { 
  11.  
  12. System.out.println("before"); 
  13.  
  14. mVendor.sell(); 
  15.  
  16. System.out.println("after"); 
  17.  
  18.  
  19. public void ad() { 
  20.  
  21. System.out.println("before"); 
  22.  
  23. mVendor.ad(); 
  24.  
  25. System.out.println("after"); 
  26.  
  27.  

從以上代碼中我們可以了解到,通過靜態代理實現我們的需求需要我們在每個方法中都添加相應的邏輯,這里只存在兩個方法所以工作量還不算大,假如Sell接口中包含上百個方法呢?這時候使用靜態代理就會編寫許多冗余代碼。通過使用動態代理,我們可以做一個“統一指示”,從而對所有代理類的方法進行統一處理,而不用逐一修改每個方法。下面我們來具體介紹下如何使用動態代理方式實現我們的需求。

2. 使用動態代理

(1)InvocationHandler接口

在使用動態代理時,我們需要定義一個位于代理類與委托類之間的中介類,這個中介類被要求實現InvocationHandler接口,這個接口的定義如下:

 

  1. public interface InvocationHandler { 
  2.  
  3. Object invoke(Object proxy, Method method, Object[] args); 
  4.  

從InvocationHandler這個名稱我們就可以知道,實現了這個接口的中介類用做“調用處理器”。當我們調用代理類對象的方法時,這個“調用”會轉送到invoke方法中,代理類對象作為proxy參數傳入,參數method標識了我們具體調用的是代理類的哪個方法,args為這個方法的參數。這樣一來,我們對代理類中的所有方法的調用都會變為對invoke的調用,這樣我們可以在invoke方法中添加統一的處理邏輯(也可以根據method參數對不同的代理類方法做不同的處理)。因此我們只需在中介類的invoke方法實現中輸出“before”,然后調用委托類的invoke方法,再輸出“after”。下面我們來一步一步具體實現它。

(2)委托類的定義

動態代理方式下,要求委托類必須實現某個接口,這里我們實現的是Sell接口。委托類Vendor類的定義如下:

 

  1. public class Vendor implements Sell { 
  2.  
  3. public void sell() { 
  4.  
  5. System.out.println("In sell method"); 
  6.  
  7.  
  8. public void ad() { 
  9.  
  10. System,out.println("ad method"
  11.  
  12.  

(3)中介類

上面我們提到過,中介類必須實現InvocationHandler接口,作為調用處理器”攔截“對代理類方法的調用。中介類的定義如下:

 

  1. public class DynamicProxy implements InvocationHandler { 
  2.  
  3. private Object obj; //obj為委托類對象; 
  4.  
  5. public DynamicProxy(Object obj) { 
  6.  
  7. this.obj = obj; 
  8.  
  9.  
  10. @Override 
  11.  
  12. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
  13.  
  14. System.out.println("before"); 
  15.  
  16. Object result = method.invoke(obj, args); 
  17.  
  18. System.out.println("after"); 
  19.  
  20. return result; 
  21.  
  22.  

從以上代碼中我們可以看到,中介類持有一個委托類對象引用,在invoke方法中調用了委托類對象的相應方法(第11行),看到這里是不是覺得似曾相識?通過聚合方式持有委托類對象引用,把外部對invoke的調用最終都轉為對委托類對象的調用。這不就是我們上面介紹的靜態代理的一種實現方式嗎?實際上,中介類與委托類構成了靜態代理關系,在這個關系中,中介類是代理類,委托類就是委托類; 代理類與中介類也構成一個靜態代理關系,在這個關系中,中介類是委托類,代理類是代理類。也就是說,動態代理關系由兩組靜態代理關系組成,這就是動態代理的原理。下面我們來介紹一下如何”指示“以動態生成代理類。

(4)動態生成代理類

動態生成代理類的相關代碼如下:

 

  1. public class Main { 
  2.  
  3. public static void main(String[] args) { 
  4.  
  5. //創建中介類實例 
  6.  
  7. DynamicProxy inter = new DynamicProxy(new Vendor()); 
  8.  
  9. //加上這句將會產生一個$Proxy0.class文件,這個文件即為動態生成的代理類文件 
  10.  
  11. System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true"); 
  12.  
  13. //獲取代理類實例sell 
  14.  
  15. Sell sell = (Sell)(Proxy.newProxyInstance(Sell.class.getClassLoader(), new Class[] {Sell.class}, inter)); 
  16.  
  17. //通過代理類對象調用代理類方法,實際上會轉到invoke方法調用 
  18.  
  19. sell.sell(); 
  20.  
  21. sell.ad(); 
  22.  
  23.  

在以上代碼中,我們調用Proxy類的newProxyInstance方法來獲取一個代理類實例。這個代理類實現了我們指定的接口并且會把方法調用分發到指定的調用處理器。這個方法的聲明如下:

 

  1. public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException 

方法的三個參數含義分別如下:

loader:定義了代理類的ClassLoder;

interfaces:代理類實現的接口列表

h:調用處理器,也就是我們上面定義的實現了InvocationHandler接口的類實例

我們運行一下,看看我們的動態代理是否能正常工作。我這里運行后的輸出為:

 

說明我們的動態代理確實奏效了。

上面我們已經簡單提到過動態代理的原理,這里再簡單的總結下:首先通過newProxyInstance方法獲取代理類實例,而后我們便可以通過這個代理類實例調用代理類的方法,對代理類的方法的調用實際上都會調用中介類(調用處理器)的invoke方法,在invoke方法中我們調用委托類的相應方法,并且可以添加自己的處理邏輯。

責任編輯:趙立京 來源: 51CTO
相關推薦

2019-04-01 14:59:56

負載均衡服務器網絡

2020-11-26 14:05:39

C ++運算符數據

2020-12-17 06:48:21

SQLkafkaMySQL

2022-03-23 09:32:38

微服務容器Kubernetes

2020-09-27 14:41:37

C語言編程語言計算機

2024-06-19 09:58:29

2022-06-16 07:31:41

Web組件封裝HTML 標簽

2021-09-07 09:40:20

Spark大數據引擎

2023-04-12 11:18:51

甘特圖前端

2016-01-04 11:18:00

KubernetesKubernetes概容器技術

2012-07-10 01:22:32

PythonPython教程

2024-05-13 09:28:43

Flink SQL大數據

2015-09-06 09:22:24

框架搭建快速高效app

2023-11-30 10:21:48

虛擬列表虛擬列表工具庫

2023-11-09 14:44:27

Docker鏡像容器

2021-03-03 11:36:57

Java 8Java 15Java

2021-07-01 06:47:30

Java泛型泛型擦除

2019-09-16 09:14:51

2009-10-09 14:45:29

VB程序

2022-08-26 09:01:07

CSSFlex 布局
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美激情第一区 | 国产福利在线 | 一区二区三区免费 | 国产91中文| 日韩欧美中文在线 | 国产夜恋视频在线观看 | 一级免费在线视频 | 久久丝袜 | 伊人伊成久久人综合网站 | 亚洲视频中文字幕 | 国产精品国产三级国产aⅴ中文 | 久久国产精品一区二区三区 | 日韩国产黄色片 | 国产一区二区在线观看视频 | 亚洲一区二区av | 日韩在线小视频 | 久久久看 | 丁香婷婷成人 | 日本高清视频在线播放 | 久久久做 | 成人美女免费网站视频 | 国产精品一区二区福利视频 | 国产成人高清在线观看 | 99亚洲| 国产在线一区二区三区 | 国产资源网| 日韩一级二级片 | 国产精品一区二区久久精品爱微奶 | 国产片网站 | 99在线免费视频 | 精品国产乱码久久久久久闺蜜 | 亚洲色图插插插 | 久久国产一区二区三区 | 午夜精品久久久久久久久久久久 | 欧美色综合天天久久综合精品 | 日韩成人在线观看 | 97精品一区二区 | 欧美日韩国产精品一区 | 中文字幕国产精品 | 国产精品资源在线 | 91精品国产色综合久久 |