Java 動(dòng)態(tài)代理與CGLIB詳細(xì)介紹
靜態(tài)代理模式
因?yàn)樾枰獙σ恍┖瘮?shù)進(jìn)行二次處理,或是某些函數(shù)不讓外界知道時(shí),可以使用代理模式,通過訪問第三方,間接訪問原函數(shù)的方式,達(dá)到以上目的。
interface Hosee{
String sayhi();
}
class Hoseeimpl implements Hosee{
@Override
public String sayhi()
{
return "Welcome oschina hosee's blog";
}
}
class HoseeProxy implements Hosee{
Hosee h;
public HoseeProxy(Hosee h)
{
this.h = h;
}
@Override
public String sayhi()
{
System.out.println("I'm proxy!");
return h.sayhi();
}
}
public class StaticProxy
{
public static void main(String[] args)
{
Hoseeimpl h = new Hoseeimpl();
HoseeProxy hp = new HoseeProxy(h);
System.out.println(hp.sayhi());
}
}
1.1 靜態(tài)代理的弊端
如果要想為多個(gè)類進(jìn)行代理,則需要建立多個(gè)代理類,維護(hù)難度加大。
仔細(xì)想想,為什么靜態(tài)代理會(huì)有這些問題,是因?yàn)榇碓诰幾g期就已經(jīng)決定,如果代理哪個(gè)發(fā)生在運(yùn)行期,這些問題解決起來就比較簡單,所以動(dòng)態(tài)代理的存在就很有必要了。
2.動(dòng)態(tài)代理
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
interface HoseeDynamic
{
String sayhi();
}
class HoseeDynamicimpl implements HoseeDynamic
{
@Override
public String sayhi()
{
return "Welcome oschina hosee's blog";
}
}
class MyProxy implements InvocationHandler
{
Object obj;
public Object bind(Object obj)
{
this.obj = obj;
return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable
{
System.out.println("I'm proxy!");
Object res = method.invoke(obj, args);
return res;
}
}
public class DynamicProxy
{
public static void main(String[] args)
{
MyProxy myproxy = new MyProxy();
HoseeDynamicimpl dynamicimpl = new HoseeDynamicimpl();
HoseeDynamic proxy = (HoseeDynamic)myproxy.bind(dynamicimpl);
System.out.println(proxy.sayhi());
}
}
類比靜態(tài)代理,可以發(fā)現(xiàn),代理類不需要實(shí)現(xiàn)原接口了,而是實(shí)現(xiàn)InvocationHandler。通過
Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
.getClass().getInterfaces(), this);
來動(dòng)態(tài)生成一個(gè)代理類,該類的類加載器與被代理類相同,實(shí)現(xiàn)的接口與被代理類相同。
通過上述方法生成的代理類相當(dāng)于靜態(tài)代理中的代理類。
這樣就實(shí)現(xiàn)了在運(yùn)行期才決定代理對象是怎么樣的,解決了靜態(tài)代理的弊端。
當(dāng)動(dòng)態(tài)生成的代理類調(diào)用方法時(shí),會(huì)觸發(fā)invoke方法,在invoke方法中可以對被代理類的方法進(jìn)行增強(qiáng)。
通過動(dòng)態(tài)代理可以很明顯的看到它的好處,在使用靜態(tài)代理時(shí),如果不同接口的某些類想使用代理模式來實(shí)現(xiàn)相同的功能,將要實(shí)現(xiàn)多個(gè)代理類,但在動(dòng)態(tài)代理中,只需要一個(gè)代理類就好了。
除了省去了編寫代理類的工作量,動(dòng)態(tài)代理實(shí)現(xiàn)了可以在原始類和接口還未知的時(shí)候,就確定代理類的代理行為,當(dāng)代理類與原始類脫離直接聯(lián)系后,就可以很靈活地重用于不同的應(yīng)用場景中。
2.1 動(dòng)態(tài)代理的弊端
代理類和委托類需要都實(shí)現(xiàn)同一個(gè)接口。也就是說只有實(shí)現(xiàn)了某個(gè)接口的類可以使用Java動(dòng)態(tài)代理機(jī)制。但是,事實(shí)上使用中并不是遇到的所有類都會(huì)給你實(shí)現(xiàn)一個(gè)接口。因此,對于沒有實(shí)現(xiàn)接口的類,就不能使用該機(jī)制。
而CGLIB則可以實(shí)現(xiàn)對類的動(dòng)態(tài)代理
2.2 回調(diào)函數(shù)原理
上文說了,當(dāng)動(dòng)態(tài)生成的代理類調(diào)用方法時(shí),會(huì)觸發(fā)invoke方法。
很顯然invoke方法并不是顯示調(diào)用的,它是一個(gè)回調(diào)函數(shù),那么回調(diào)函數(shù)是怎么被調(diào)用的呢?
上述動(dòng)態(tài)代理的代碼中,唯一不清晰的地方只有
Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
.getClass().getInterfaces(), this);
跟蹤這個(gè)方法的源碼,可以看到程序進(jìn)行了驗(yàn)證、優(yōu)化、緩存、同步、生成字節(jié)碼、顯示類加載等操作,前面的步驟并不是我們關(guān)注的重點(diǎn),而最后它調(diào)用了
byte[] proxyClassFile = ProxyGenerator.generateProxyClass(
proxyName, interfaces);
該方法用來完成生成字節(jié)碼的動(dòng)作,這個(gè)方法可以在運(yùn)行時(shí)產(chǎn)生一個(gè)描述代理類的字節(jié)碼byte[]數(shù)組。
在main函數(shù)中加入
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");
加入這句代碼后再次運(yùn)行程序,磁盤中將會(huì)產(chǎn)生一個(gè)名為”$Proxy().class”的代理類Class文件,反編譯(反編譯工具我使用的是 JD-GUI )后可以看見如下代碼:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
public final class $Proxy0 extends Proxy
implements HoseeDynamic
{
private static Method m1;
private static Method m3;
private static Method m0;
private static Method m2;
public $Proxy0(InvocationHandler paramInvocationHandler)
throws
{
super(paramInvocationHandler);
}
public final boolean equals(Object paramObject)
throws
{
try
{
return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final String sayhi()
throws
{
try
{
return (String)this.h.invoke(this, m3, null);
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final int hashCode()
throws
{
try
{
return ((Integer)this.h.invoke(this, m0, null)).intValue();
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final String toString()
throws
{
try
{
return (String)this.h.invoke(this, m2, null);
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
static
{
try
{
m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
m3 = Class.forName("HoseeDynamic").getMethod("sayhi", new Class[0]);
m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
return;
}
catch (NoSuchMethodException localNoSuchMethodException)
{
throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
}
catch (ClassNotFoundException localClassNotFoundException)
{
throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
}
}
}
動(dòng)態(tài)代理類不僅代理了顯示定義的接口中的方法,而且還代理了java的根類Object中的繼承而來的equals()、hashcode()、toString()這三個(gè)方法,并且僅此三個(gè)方法。
可以在上述代碼中看到,無論調(diào)用哪個(gè)方法,都會(huì)調(diào)用到InvocationHandler的invoke方法,只是參數(shù)不同。
2.3 動(dòng)態(tài)代理與靜態(tài)代理的區(qū)別
Proxy類的代碼被固定下來,不會(huì)因?yàn)闃I(yè)務(wù)的逐漸龐大而龐大;
可以實(shí)現(xiàn)AOP編程,這是靜態(tài)代理無法實(shí)現(xiàn)的;
解耦,如果用在web業(yè)務(wù)下,可以實(shí)現(xiàn)數(shù)據(jù)層和業(yè)務(wù)層的分離。
動(dòng)態(tài)代理的優(yōu)勢就是實(shí)現(xiàn)無侵入式的代碼擴(kuò)展。 靜態(tài)代理這個(gè)模式本身有個(gè)大問題,如果類方法數(shù)量越來越多的時(shí)候,代理類的代碼量是十分龐大的。所以引入動(dòng)態(tài)代理來解決此類問題
3. CGLIB
cglib是針對類來實(shí)現(xiàn)代理的,他的原理是對指定的目標(biāo)類生成一個(gè)子類,并覆蓋其中方法實(shí)現(xiàn)增強(qiáng),但因?yàn)椴捎玫氖抢^承,所以不能對final修飾的類進(jìn)行代理。
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
class CGlibHosee
{
public String sayhi()
{
return "Welcome oschina hosee's blog";
}
}
class CGlibHoseeProxy
{
Object obj;
public Object bind(final Object target)
{
this.obj = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(obj.getClass());
enhancer.setCallback(new MethodInterceptor()
{
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable
{
System.out.println("I'm proxy!");
Object res = method.invoke(target, args);
return res;
}
});
return enhancer.create();
}
}
public class CGlibProxy
{
public static void main(String[] args)
{
CGlibHosee cGlibHosee = new CGlibHosee();
CGlibHoseeProxy cGlibHoseeProxy = new CGlibHoseeProxy();
CGlibHosee proxy = (CGlibHosee) cGlibHoseeProxy.bind(cGlibHosee);
System.out.println(proxy.sayhi());
}
}
cglib需要指定父類和回調(diào)方法。當(dāng)然cglib也可以與Java動(dòng)態(tài)代理一樣面向接口,因?yàn)楸举|(zhì)是繼承。
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
- Java基礎(chǔ)之動(dòng)態(tài)代理Cglib詳解
- 詳解Java Cglib動(dòng)態(tài)代理
- Java使用JDK與Cglib動(dòng)態(tài)代理技術(shù)統(tǒng)一管理日志記錄
- JAVA中的靜態(tài)代理、動(dòng)態(tài)代理以及CGLIB動(dòng)態(tài)代理總結(jié)
- Java CGLib動(dòng)態(tài)代理機(jī)制(全面解析)
- java 中動(dòng)態(tài)代理(JDK,cglib)實(shí)例代碼
- 深入理解java動(dòng)態(tài)代理的兩種實(shí)現(xiàn)方式(JDK/Cglib)
- 淺談Java代理(jdk靜態(tài)代理、動(dòng)態(tài)代理和cglib動(dòng)態(tài)代理)
- java動(dòng)態(tài)代理和cglib動(dòng)態(tài)代理示例分享
- java動(dòng)態(tài)代理(jdk與cglib)詳細(xì)解析
- Java cglib動(dòng)態(tài)代理原理分析
相關(guān)文章
使用JSCH框架通過跳轉(zhuǎn)機(jī)訪問其他節(jié)點(diǎn)的方法
下面小編就為大家分享一篇使用JSCH框架通過跳轉(zhuǎn)機(jī)訪問其他節(jié)點(diǎn)的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2017-12-12
解決IDEA2020 創(chuàng)建maven項(xiàng)目沒有src/main/java目錄和webapp目錄問題
這篇文章主要介紹了IDEA2020 創(chuàng)建maven項(xiàng)目沒有src/main/java目錄和webapp目錄問題解決方法,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10
淺析Java中靜態(tài)代理和動(dòng)態(tài)代理的應(yīng)用與區(qū)別
代理模式在我們生活中很常見,而Java中常用的兩個(gè)的代理模式就是動(dòng)態(tài)代理與靜態(tài)代理,這篇文章主要為大家介紹了二者的應(yīng)用與區(qū)別,需要的可以參考下2023-08-08
Java操作Elasticsearch?rest-high-level-client?的基本使用
這篇文章主要介紹了Java操作Elasticsearch?rest-high-level-client?的基本使用,本篇主要講解一下?rest-high-level-client?去操作?Elasticsearch的方法,結(jié)合實(shí)例代碼給大家詳細(xì)講解,需要的朋友可以參考下2022-10-10
Spring中@Autowired與@Resource的區(qū)別詳析
@Autowired與@Resource都可以用來裝配bean,都可以寫在字段上,或?qū)懺趕etter方法上,下面這篇文章主要給大家介紹了關(guān)于Spring中@Autowired與@Resource區(qū)別的相關(guān)資料,需要的朋友可以參考下2021-10-10

