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

来源:https://www.cnblogs.com/hellohowlow/archive/2018/02/26/8476227.html
-Advertisement-
Play Games

十分鐘幫助大家理解Java中的動態代理,什麼是動態代理?感興趣的小伙伴們可以參考一下 十分鐘幫助大家理解Java中的動態代理,什麼是動態代理?感興趣的小伙伴們可以參考一下 若代理類在程式運行前就已經存在,那麼這種代理方式被成為 靜態代理 ,這種情況下的代理類通常都是我們在Java代碼中定義的。 通常 ...


十分鐘幫助大家理解Java中的動態代理,什麼是動態代理?感興趣的小伙伴們可以參考一下  

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

一、概述
1. 什麼是代理
我們大家都知道微商代理,簡單地說就是代替廠家賣商品,廠家“委托”代理為其銷售商品。關於微商代理,首先我們從他們那裡買東西時通常不知道背後的廠家究竟是誰,也就是說,“委托者”對我們來說是不可見的;其次,微商代理主要以朋友圈的人為目標客戶,這就相當於為廠家做了一次對客戶群體的“過濾”。我們把微商代理和廠家進一步抽象,前者可抽象為代理類,後者可抽象為委托類(被代理類)。通過使用代理,通常有兩個優點,並且能夠分別與我們提到的微商代理的兩個特點對應起來:
優點一:可以隱藏委托類的實現;
優點二:可以實現客戶與委托類間的解耦,在不修改委托類代碼的情況下能夠做一些額外的處理。
2. 靜態代理
若代理類在程式運行前就已經存在,那麼這種代理方式被成為 靜態代理 ,這種情況下的代理類通常都是我們在Java代碼中定義的。 通常情況下, 靜態代理中的代理類和委托類會實現同一介面或是派生自相同的父類。 下麵我們用Vendor類代表生產廠家,BusinessAgent類代表微商代理,來介紹下靜態代理的簡單實現,委托類和代理類都實現了Sell介面,Sell介面的定義如下:

public interface Sell { void sell(); void ad(); 
 
} 
Vendor類的定義如下:
public class Vendor implements Sell { public void sell() { 
 
System.out.println("In sell method"); 
 
} public void ad() { 
 
System,out.println("ad method") 
 
} 
 
} 

代理類BusinessAgent的定義如下:

public class Vendor implements Sell { public void sell() { 
 
System.out.println("In sell method"); 
 
} public void ad() { 
 
System,out.println("ad method") 
 
} 
 
} 

從BusinessAgent類的定義我們可以瞭解到,靜態代理可以通過聚合來實現,讓代理類持有一個委托類的引用即可。
下麵我們考慮一下這個需求:給Vendor類增加一個過濾功能,只賣貨給大學生。通過靜態代理,我們無需修改Vendor類的代碼就可以實現,只需在BusinessAgent類中的sell方法中添加一個判斷即可如下所示:

public class BusinessAgent implements Sell { 
 
... 
 
public void sell() { 
 
if (isCollegeStudent()) { 
 
vendor.sell(); 
 
} 
 
} 
 
... 
 
} 

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

二、動態代理
1. 什麼是動態代理
代理類在程式運行時創建的代理方式被成為 動態代理。 也就是說,這種情況下,代理類並不是在Java代碼中定義的,而是在運行時根據我們在Java代碼中的“指示”動態生成的。相比於靜態代理, 動態代理的優勢在於可以很方便的對代理類的函數進行統一的處理,而不用修改每個代理類的函數。 這麼說比較抽象,下麵我們結合一個實例來介紹一下動態代理的這個優勢是怎麼體現的。
現在,假設我們要實現這樣一個需求:在執行委托類中的方法之前輸出“before”,在執行完畢後輸出“after”。我們還是以上面例子中的Vendor類作為委托類,BusinessAgent類作為代理類來進行介紹。首先我們來使用靜態代理來實現這一需求,相關代碼如下:

public class BusinessAgent implements Sell { 
 
private Vendor mVendor; 
 
public BusinessAgent(Vendor vendor) { 
 
this.mVendor = vendor; 
 
} 
 
public void sell() { 
 
System.out.println("before"); 
 
mVendor.sell(); 
 
System.out.println("after"); 
 
} 
 
public void ad() { 
 
System.out.println("before"); 
 
mVendor.ad(); 
 
System.out.println("after"); 
 
} 
 
} 

從以上代碼中我們可以瞭解到,通過靜態代理實現我們的需求需要我們在每個方法中都添加相應的邏輯,這裡只存在兩個方法所以工作量還不算大,假如Sell介面中包含上百個方法呢?這時候使用靜態代理就會編寫許多冗餘代碼。通過使用動態代理,我們可以做一個“統一指示”,從而對所有代理類的方法進行統一處理,而不用逐一修改每個方法。下麵我們來具體介紹下如何使用動態代理方式實現我們的需求。
2. 使用動態代理
(1)InvocationHandler介面
在使用動態代理時,我們需要定義一個位於代理類與委托類之間的中介類,這個中介類被要求實現InvocationHandler介面,這個介面的定義如下:

public interface InvocationHandler { 
 
Object invoke(Object proxy, Method method, Object[] args); 
 
} 

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

public class Vendor implements Sell { 
 
public void sell() { 
 
System.out.println("In sell method"); 
 
} 
 
public void ad() { 
 
System,out.println("ad method") 
 
} 
 
} 

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

public class DynamicProxy implements InvocationHandler { 
 
private Object obj; //obj為委托類對象; 
 
public DynamicProxy(Object obj) { 
 
this.obj = obj; 
 
} 
 
@Override 
 
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
 
System.out.println("before"); 
 
Object result = method.invoke(obj, args); 
 
System.out.println("after"); 
 
return result; 
 
} 
 
} 

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

public class Main { 
 
public static void main(String[] args) { 
 
//創建中介類實例 
 
DynamicProxy inter = new DynamicProxy(new Vendor()); 
 
//加上這句將會產生一個$Proxy0.class文件,這個文件即為動態生成的代理類文件 
 
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true"); 
 
//獲取代理類實例sell 
 
Sell sell = (Sell)(Proxy.newProxyInstance(Sell.class.getClassLoader(), new Class[] {Sell.class}, inter)); 
 
//通過代理類對象調用代理類方法,實際上會轉到invoke方法調用 
 
sell.sell(); 
 
sell.ad(); 
 
} 
 
} 

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

複製代碼代碼如下: public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException
方法的三個參數含義分別如下:
loader:定義了代理類的ClassLoder;
interfaces:代理類實現的介面列表
h:調用處理器,也就是我們上面定義的實現了InvocationHandler介面的類實例
我們運行一下,看看我們的動態代理是否能正常工作。我這裡運行後的輸出為:

 

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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C++ 能夠使用流提取運算符 和流插入運算符 和插入運算符 using namespace std; class Person{ public: Person(const char str) : name(str){} int GetAge(){ return this age; } / 聲明為類的 ...
  • 1.協程(微線程)協程是一種用戶態的輕量級線程。協程擁有自己的寄存器上下文和棧。協程調度切換時,將寄存器上下文和棧保存到其他地方,在切回來的時候,恢復先前保存的寄存器上下文和棧。因此: 協程能保留上一次調用時的狀態(即所有局部狀態的一個特定組合),每次過程重入時,就相當於進入上一次調用的狀態,換種說 ...
  • jdk1.8.0_144 Map是Java三種集合中的一種位於java.util包中,Map作為一個介面存在定義了這種數據結構的一些基礎操作,它的最終實現類有很多:HashMap、TreeMap、SortedMap等等,這些最終的子類大多有一個共同的抽象父類AbstractMap。在Abstract ...
  • 1.隊列(queue) 用法: 作用: 1)解耦 2)提高效率 class queue.Queue(maxsize=0) #先入先出class queue.LifoQueue(maxsize=0) #後進先出 class queue.PriorityQueue(maxsize=0) #存儲數據時可設 ...
  • 上篇博文對Spring的工作原理做了個大概的介紹,想看的同學請出門左轉。今天詳細說幾點。 (一)Spring IoC容器及其實例化與使用 Spring IoC容器負責Bean的實例化、配置和組裝工作有兩個介面:BeanFactory和ApplicationContext。其中ApplicationC ...
  • 個人對selenium的理解: 1.使用selenium操作瀏覽器,實際上是使用selenium框架下的webdriver啟動各瀏覽器的驅動實現對瀏覽器的操作的。以下兩句代碼即可啟動firefox瀏覽器驅動 from selenium import webdriver brower=webdrive ...
  • 以前寫過一個MVC框架,封裝的有點low,經過一段時間的沉澱,打算重新改造下,之前這篇文章封裝過一個驗證碼類。 這次重新改造MVC有幾個很大的收穫 >全部代碼都是用Ubuntu+Vim編寫,以前都是windows上開發,這次徹底迷上Ubuntu Linux >裸裝php,用php自帶的伺服器解釋執行 ...
  • 原理:根據IP,資源ID,時間戳,一次性Access_Token,APPKEY(暴露在前臺)和APPSERECT(後臺)來生成參數,具體見下麵: 瀏覽器請求頁面=>後臺引用防盜鏈代碼=>生成Access_Token並綁定訪問IP=>生成sign=>生成網址 如:原圖片頁面為http://xxx.ex ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...