溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java CGLib動(dòng)態(tài)代理機(jī)制(全面解析)

發(fā)布時(shí)間:2020-09-23 00:47:18 來(lái)源:腳本之家 閱讀:152 作者:jingxian 欄目:編程語(yǔ)言

一、首先說(shuō)一下JDK中的動(dòng)態(tài)代理:

JDK中的動(dòng)態(tài)代理是通過(guò)反射類Proxy以及InvocationHandler回調(diào)接口實(shí)現(xiàn)的

但是,JDK中所要進(jìn)行動(dòng)態(tài)代理的類必須要實(shí)現(xiàn)一個(gè)接口,也就是說(shuō)只能對(duì)該類所實(shí)現(xiàn)接口中定義的方法進(jìn)行代理,這在實(shí)際編程中具有一定的局限性,而且使用反射的效率也并不是很高。

二、使用CGLib實(shí)現(xiàn):

使用CGLib實(shí)現(xiàn)動(dòng)態(tài)代理,完全不受代理類必須實(shí)現(xiàn)接口的限制,而且CGLib底層采用ASM字節(jié)碼生成框架,使用字節(jié)碼技術(shù)生成代理類,比使用Java反射效率要高。唯一需要注意的是,CGLib不能對(duì)聲明為final的方法進(jìn)行代理,因?yàn)镃GLib原理是動(dòng)態(tài)生成被代理類的子類。

下面,將通過(guò)一個(gè)實(shí)例介紹使用CGLib實(shí)現(xiàn)動(dòng)態(tài)代理。

1、被代理類:

首先,定義一個(gè)類,該類沒(méi)有實(shí)現(xiàn)任何接口,包含兩個(gè)方法。

public class ConcreteClassNoInterface { 
 public String getConcreteMethodA(String str){ 
  System.out.println("ConcreteMethod A ... "+str); 
  return str; 
 } 
 public int getConcreteMethodB(int n){ 
  System.out.println("ConcreteMethod B ... "+n); 
  return n+10; 
 } 
}

2、攔截器:

定義一個(gè)攔截器。在調(diào)用目標(biāo)方法時(shí),CGLib會(huì)回調(diào)MethodInterceptor接口方法攔截,來(lái)實(shí)現(xiàn)你自己的代理邏輯,類似于JDK中的InvocationHandler接口。

public class ConcreteClassInterceptor implements MethodInterceptor{ 
 public Object intercept(Object obj, Method method, Object[] arg, MethodProxy proxy) throws Throwable { 
  System.out.println("Before:"+method); 
  Object object=proxy.invokeSuper(obj, arg); 
  System.out.println("After:"+method); 
  return object; 
 } 
} 

參數(shù):Object為由CGLib動(dòng)態(tài)生成的代理類實(shí)例,Method為上文中實(shí)體類所調(diào)用的被代理的方法引用,Object[]為參數(shù)值列表,MethodProxy為生成的代理類對(duì)方法的代理引用。

返回:從代理實(shí)例的方法調(diào)用返回的值。

其中,proxy.invokeSuper(obj,arg):

調(diào)用代理類實(shí)例上的proxy方法的父類方法(即實(shí)體類ConcreteClassNoInterface中對(duì)應(yīng)的方法)

在這個(gè)示例中,只在調(diào)用被代理類方法前后各打印了一句話,當(dāng)然實(shí)際編程中可以是其它復(fù)雜邏輯。

3、生成動(dòng)態(tài)代理類:

Enhancer enhancer=new Enhancer(); 
enhancer.setSuperclass(ConcreteClassNoInterface.class); 
enhancer.setCallback(new ConcreteClassInterceptor()); 
ConcreteClassNoInterface ccni=(ConcreteClassNoInterface)enhancer.create(); 

這里Enhancer類是CGLib中的一個(gè)字節(jié)碼增強(qiáng)器,它可以方便的對(duì)你想要處理的類進(jìn)行擴(kuò)展,以后會(huì)經(jīng)??吹剿?/p>

首先將被代理類ConcreteClassNoInterface設(shè)置成父類,然后設(shè)置攔截器ConcreteClassInterceptor,最后執(zhí)行enhancer.create()動(dòng)態(tài)生成一個(gè)代理類,并從Object強(qiáng)制轉(zhuǎn)型成父類型ConcreteClassNoInterface。

最后,在代理類上調(diào)用方法:

ccni.getConcreteMethodA("shensy"); 
ccni.getConcreteMethodB(0); 

查看控制臺(tái)輸出:

Before :public java.lang.String generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodA(java.lang.String) 
ConcreteMethod A ... shensy 
After :public java.lang.String generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodA(java.lang.String) 
Before :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int) 
ConcreteMethod B ... 0 
After :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int) 

可以看到,攔截器在調(diào)用被代理類方法前后都執(zhí)行了print操作。

以上這篇Java CGLib動(dòng)態(tài)代理機(jī)制(全面解析)就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI