java動(dòng)態(tài)代理和cglib動(dòng)態(tài)代理示例分享
java動(dòng)態(tài)代理類可以分為兩種。
靜態(tài)代理:由程序員創(chuàng)建或特定工具自動(dòng)生成源代碼,再對(duì)其編譯。在程序運(yùn)行前,代理類的.class文件就已經(jīng)存在了。
動(dòng)態(tài)代理:在程序運(yùn)行時(shí),運(yùn)用反射機(jī)制動(dòng)態(tài)創(chuàng)建而成。
一、首先我們進(jìn)行java動(dòng)態(tài)代理的演示。
現(xiàn)在我們有一個(gè)簡(jiǎn)單的業(yè)務(wù)接口Saying,如下:
package testAOP;
public interface Saying {
public void sayHello(String name);
public void talking(String name);
}
一個(gè)簡(jiǎn)單的實(shí)現(xiàn)類SayingImpl,如下:
package testAOP;
public class SayingImpl implements Saying {
@Override
public void sayHello(String name) {
// TODO Auto-generated method stub
System.out.println(name + ":大家好?。?);
}
@Override
public void talking(String name) {
// TODO Auto-generated method stub
System.out.println(name + ":我的意思是,我們要努力建設(shè)和諧社會(huì)!");
}
}
我們要實(shí)現(xiàn)的是,在sayHello和talking之前和之后分別動(dòng)態(tài)植入處理。
JDK動(dòng)態(tài)代理主要用到j(luò)ava.lang.reflect包中的兩個(gè)類:Proxy和InvocationHandler.
InvocationHandler是一個(gè)接口,通過實(shí)現(xiàn)該接口定義橫切邏輯,并通過反射機(jī)制調(diào)用目標(biāo)類的代碼,動(dòng)態(tài)的將橫切邏輯和業(yè)務(wù)邏輯編織在一起。
Proxy利用InvocationHandler動(dòng)態(tài)創(chuàng)建一個(gè)符合某一接口的實(shí)例,生成目標(biāo)類的代理對(duì)象。
如下,我們創(chuàng)建一個(gè)InvocationHandler實(shí)例:
package testAOP;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class MyInvocationHandler implements InvocationHandler {
private Object target;
MyInvocationHandler(Object target){
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
//目標(biāo)方法前執(zhí)行
System.out.println("——————————————————————————");
System.out.println("下一位請(qǐng)登臺(tái)發(fā)言!");
//目標(biāo)方法調(diào)用
Object obj = method.invoke(target, args);
//目標(biāo)方法后執(zhí)行
System.out.println("大家掌聲鼓勵(lì)!");
return obj;
}
}
下面是測(cè)試:
package testAOP;
import java.lang.reflect.Proxy;
public class JDKProxyTest {
public static void main(String[] args) {
// 希望被代理的目標(biāo)業(yè)務(wù)類
Saying target = new SayingImpl();
// 將目標(biāo)類和橫切類編織在一起
MyInvocationHandler handler = new MyInvocationHandler(target);
// 創(chuàng)建代理實(shí)例
Saying proxy = (Saying) Proxy.newProxyInstance(
target.getClass().getClassLoader(),//目標(biāo)類的類加載器
target.getClass().getInterfaces(),//目標(biāo)類的接口
handler);//橫切類
proxy.sayHello("小明");
proxy.talking("小麗");
}
}
運(yùn)行情況如下:
——————————————————————————
下一位請(qǐng)登臺(tái)發(fā)言!
小明:大家好?。?BR>大家掌聲鼓勵(lì)!
——————————————————————————
下一位請(qǐng)登臺(tái)發(fā)言!
小麗:我的意思是,我們要努力建設(shè)和諧社會(huì)!
大家掌聲鼓勵(lì)!
使用JDK動(dòng)態(tài)代理有一個(gè)很大的限制,就是它要求目標(biāo)類必須實(shí)現(xiàn)了對(duì)應(yīng)方法的接口,它只能為接口創(chuàng)建代理實(shí)例。我們?cè)谏衔臏y(cè)試類中的Proxy的newProxyInstance方法中可以看到,該方法第二個(gè)參數(shù)便是目標(biāo)類的接口。如果該類沒有實(shí)現(xiàn)接口,這就要靠cglib動(dòng)態(tài)代理了。
CGLib采用非常底層的字節(jié)碼技術(shù),可以為一個(gè)類創(chuàng)建一個(gè)子類,并在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用,并順勢(shì)植入橫切邏輯。
二、接下來我們進(jìn)行cglib動(dòng)態(tài)代理的演示。
首先我們需要導(dǎo)包,我用的包是cglib-nodep-2.1_3.jar。
我們首先創(chuàng)建一個(gè)代理創(chuàng)建器CglibProxy:
package testAOP.cglib;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class CglibProxy implements MethodInterceptor{
Enhancer enhancer = new Enhancer();
public Object getProxy(Class clazz) {
//設(shè)置需要?jiǎng)?chuàng)建的子類
enhancer.setSuperclass(clazz);
enhancer.setCallback(this);
//通過字節(jié)碼技術(shù)動(dòng)態(tài)創(chuàng)建子類實(shí)例
return enhancer.create();
}
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable {
// TODO Auto-generated method stub
System.out.println("——————————————————————————");
System.out.println("下一位請(qǐng)登臺(tái)發(fā)言!");
//目標(biāo)方法調(diào)用
Object result = proxy.invokeSuper(obj, args);
//目標(biāo)方法后執(zhí)行
System.out.println("大家掌聲鼓勵(lì)!");
return result;
}
}
然后進(jìn)行測(cè)試:
package testAOP.cglib;
import testAOP.Saying;
import testAOP.SayingImpl;
public class CglibProxyTest {
public static void main(String[] args) {
CglibProxy proxy = new CglibProxy();
//通過動(dòng)態(tài)生成子類的方式創(chuàng)建代理類
Saying target = (Saying) proxy.getProxy(SayingImpl.class);
target.sayHello("小明");
target.talking("小麗");
}
}
結(jié)果與JDK動(dòng)態(tài)代理沒有任何區(qū)別。
JDK動(dòng)態(tài)代理和CGLib動(dòng)態(tài)代理都是運(yùn)行時(shí)增強(qiáng),通過將橫切代碼植入代理類的方式增強(qiáng)。與此不同的是AspectJ,它能夠在通過特殊的編譯器在編譯時(shí)期將橫切代碼植入增強(qiáng),這樣的增強(qiáng)處理在運(yùn)行時(shí)候更有優(yōu)勢(shì),因?yàn)镴DK動(dòng)態(tài)代理和CGLib動(dòng)態(tài)代理每次運(yùn)行都需要增強(qiáng)。
- 十分鐘理解Java中的動(dòng)態(tài)代理
- 代理模式之Java動(dòng)態(tài)代理實(shí)現(xiàn)方法
- java代理模式與動(dòng)態(tài)代理模式詳解
- Java中反射動(dòng)態(tài)代理接口的詳解及實(shí)例
- 基于接口實(shí)現(xiàn)java動(dòng)態(tài)代理示例
- 詳解java中動(dòng)態(tài)代理實(shí)現(xiàn)機(jī)制
- java實(shí)現(xiàn)動(dòng)態(tài)代理方法淺析
- 詳解java動(dòng)態(tài)代理的2種實(shí)現(xiàn)方式
- Java動(dòng)態(tài)代理的應(yīng)用詳解
- Java實(shí)現(xiàn)動(dòng)態(tài)代理的實(shí)例代碼
相關(guān)文章
SpringBoot關(guān)于自動(dòng)注入mapper為空的坑及解決
這篇文章主要介紹了SpringBoot關(guān)于自動(dòng)注入mapper為空的坑及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-07-07
剖析Java中在Collection集合中使用contains和remove為什么要重寫equals
這篇文章主要介紹了Collection集合的contains和remove方法詳解remove以及相關(guān)的經(jīng)驗(yàn)技巧,通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09
Java中bcrypt算法實(shí)現(xiàn)密碼加密的方法步驟
我們可以在Spring Boot和SSM中實(shí)現(xiàn)密碼加密,使用bcrypt算法可以保障密碼的安全性,并且減少了手動(dòng)編寫哈希函數(shù)的工作量,本文就來詳細(xì)的介紹一下,感興趣的可以了解一下2023-08-08

