JDK動態(tài)代理詳細解析
一、說明
在Java的動態(tài)代理機制中,有兩個重要的類和接口,一個是InvoInvocationHandler(接口)、Proxy(類),這一個類和接口是我們動態(tài)代理所必須用到的。
優(yōu)點:
- 對于實現了接口的類,可以直接使用基于接口的動態(tài)代理進行代理,非常方便
- 代理類和被代理類都必須實現同一個接口,能夠實現對被代理對象的方法調用進行統一管理。
- 性能上:在老版的jdk,jdk代理生成的類速度快,通過反射調用慢,cglib是jdk代理速度的10倍左右,jdk在版本每次升級都會有很大的性能提升,cglib停滯不前,jdk7 8的動態(tài)代理性能在1萬次實驗中比cglib要快20%左右
- jdk動態(tài)代理如果目標類未實現接口則無法代理,cglib是通過繼承的方式來動態(tài)代理,若目標類被final關鍵字修飾,則無法使用cglib做動態(tài)代理
缺點:
- 只能代理實現了接口的類,對于沒有實現接口的類無法使用此種方式進行代理。
- jdk動態(tài)代理只提供實現接口的目標類代理,不支持沒有實現接口的目標類的代理。如果目標類沒有實現接口,只能用cglib代理
二、主要類方法的說明
InvocationHandler接口
每個動態(tài)代理類都必須實現InvocationHandler接口,并且每個代理類的實例都關聯到了一個handler,當我們通過代理對象調用一個方法的時候,這個方法的調用就會被轉發(fā)為由InvocationHandler接口的invoke方法來進行調用。
InvocationHandler接口的invoke方法
- Object invoke(Object proxy, Method method, Object[] args) throws Throwable
- proxy: - 指代我們所代理的那個真實對象
- method: - 指代的是我們所要調用真實對象的某個方法的Method對象
- args: - 指代的是調用真實對象某個方法時接受的參數
**proxy存在的意義:**
1. 可以使用反射獲取代理對象的信息(也就是proxy.getClass().getName())。
2. 可以將代理對象返回以進行連續(xù)調用,這就是proxy存在的目的,因為this并不是代理對象。
Proxy 類
Proxy類的作用就是用來動態(tài)創(chuàng)建一個代理類對象的類,它提供了許多的方法,但是我們用的最多的就是newProxyInstance這個方法
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException
- loader: 一個ClassLoader對象,定義了由哪個ClassLoader對象來對生成的代理對象進行加載
- interfaces: 一個Interface對象的數組,表示的是我將要給我需要代理的對象提供一組什么接口,如果我提供了一組接口給它,那么這個代理對象就宣稱實現了該接口(多態(tài)),這樣我就能調用這組接口中的方法了
- h: 一個InvocationHandler對象,表示的是當我這個動態(tài)代理對象在調用方法的時候,會關聯到哪一個InvocationHandler對象上
打印代理的類:
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
三、關鍵步驟
創(chuàng)建InvocationHandler實現類
public class MapperProxy<T> implements InvocationHandler { private Class<T> proxyInterface; //這里可以維護一個緩存,存這個接口的方法抽象的對象 MapperProxy(Class<T> proxyInterface){ this.proxyInterface = proxyInterface; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("執(zhí)行前..."); Constructor constructor = proxyInterface.getConstructor(); Object o = constructor.newInstance(); method.invoke(o,args); System.out.println("執(zhí)行后"); return null; } }
通過 Proxy.newProxyInstance() 創(chuàng)建代理實例
Fly fly = (Fly) Proxy.newProxyInstance(MyFly.class.getClassLoader(),new Class[]{Fly.class, Fly.Fly2.class},new MapperProxy<>(MyFly.class));
- 創(chuàng)建代理類- Class<?> cl = getProxyClass0(loader, intfs);
- 獲取有參構造器 Constructor<?> cons = cl.getConstructor(constructorParams);這里的參數就是InvocationHandler
- 通過構造器創(chuàng)建代理實例-參數就是方法的第三個參數
分析生成的代理類
結論
- 代理類會繼承Proxy ,這里也就解釋了為什么通過JDK生成的代理無法代理非接口實現類了
- 代理類實現了傳入的所有接口類型
調用代理類的doFly()
這里會調用 super.h.invoke(this, m3, (Object[])null);
super.h就是我們再創(chuàng)建代理對象是傳入的MapperProxy,所有這里會執(zhí)行MapperProxy.invoke方法(在這里我們就可動態(tài)的對該執(zhí)行方法進行增強)
到此這篇關于JDK動態(tài)代理詳細解析的文章就介紹到這了,更多相關JDK動態(tài)代理內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
maven創(chuàng)建spark項目的pom.xml文件配置demo
這篇文章主要為大家介紹了maven創(chuàng)建spark項目的pom.xml文件配置demo,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-05-05