Java實(shí)例講解動(dòng)態(tài)代理
JDK動(dòng)態(tài)代理:利用反射機(jī)制生成一個(gè)實(shí)現(xiàn)代理接口的匿名類,在調(diào)用具體方法前調(diào)用InvokeHandler來處理。
CGlib動(dòng)態(tài)代理:利用ASM(開源的Java字節(jié)碼編輯庫,操作字節(jié)碼)開源包,將代理對(duì)象類的class文件加載進(jìn)來,通過修改其字節(jié)碼生成子類來處理。
區(qū)別:JDK代理只能對(duì)實(shí)現(xiàn)接口的類生成代理;CGlib是針對(duì)類實(shí)現(xiàn)代理,對(duì)指定的類生成一個(gè)子類,并覆蓋其中的方法,這種通過繼承類的實(shí)現(xiàn)方式,不能代理final修飾的類。
強(qiáng)制使用CGlib
<!-- proxy-target-class="false"默認(rèn)使用JDK動(dòng)態(tài)代理 --> <aop:aspectj-autoproxy proxy-target-class="true"/> <aop-config proxy-target-class="true">
具體代碼示例:
/**
* 目標(biāo)接口類
*/
public interface UserManager {
public void addUser(String id, String password);
public void delUser(String id);
}/**
* 接口實(shí)現(xiàn)類
*/
public class UserManagerImpl implements UserManager {
@Override
public void addUser(String id, String password) {
System.out.println("調(diào)用了UserManagerImpl.addUser()方法!");
}
@Override
public void delUser(String id) {
System.out.println("調(diào)用了UserManagerImpl.delUser()方法!");
}
}
/**
* JDK動(dòng)態(tài)代理類
*/
public class JDKProxy implements InvocationHandler {
// 需要代理的目標(biāo)對(duì)象
private Object targetObject;
public Object newProxy(Object targetObject) {
// 將目標(biāo)對(duì)象傳入進(jìn)行代理
this.targetObject = targetObject;
// 返回代理對(duì)象
return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);
}
// invoke方法
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 進(jìn)行邏輯處理的函數(shù)
checkPopedom();
Object ret = null;
// 調(diào)用invoke方法
ret = method.invoke(targetObject, args);
return ret;
}
private void checkPopedom() {
// 模擬檢查權(quán)限
System.out.println("檢查權(quán)限:checkPopedom()!");
}
} /**
* CGlib動(dòng)態(tài)代理類
*/
public class CGLibProxy implements MethodInterceptor {
// CGlib需要代理的目標(biāo)對(duì)象
private Object targetObject;
public Object createProxyObject(Object obj) {
this.targetObject = obj;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(obj.getClass());
enhancer.setCallback(this);
Object proxyObj = enhancer.create();
return proxyObj;
}
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
Object obj = null;
// 過濾方法
if ("addUser".equals(method.getName())) {
// 檢查權(quán)限
checkPopedom();
}
obj = method.invoke(targetObject, args);
return obj;
}
private void checkPopedom() {
System.out.println("檢查權(quán)限:checkPopedom()!");
}
}/**
* 測(cè)試類
*/
public class ProxyTest {
public static void main(String[] args) {
UserManager userManager = (UserManager)new CGLibProxy().createProxyObject(new UserManagerImpl());
System.out.println("CGLibProxy:");
userManager.addUser("tom", "root");
System.out.println("JDKProxy:");
JDKProxy jdkProxy = new JDKProxy();
UserManager userManagerJDK = (UserManager)jdkProxy.newProxy(new UserManagerImpl());
userManagerJDK.addUser("tom", "root");
}
}
// 運(yùn)行結(jié)果
CGLibProxy:
檢查權(quán)限checkPopedom()!
調(diào)用了UserManagerImpl.addUser()方法!
JDKProxy:
檢查權(quán)限checkPopedom()!
掉用了UserManagerImpl.addUser()方法!
總結(jié):
- JDK代理使用的是反射機(jī)制實(shí)現(xiàn)aop的動(dòng)態(tài)代理,CGLIB代理使用字節(jié)碼處理框架asm,通過修改字節(jié)碼生成子類。所以jdk動(dòng)態(tài)代理的方式創(chuàng)建代理對(duì)象效率較高,執(zhí)行效率較低,cglib創(chuàng)建效率較低,執(zhí)行效率高;
- JDK動(dòng)態(tài)代理機(jī)制是委托機(jī)制,具體說動(dòng)態(tài)實(shí)現(xiàn)接口類,在動(dòng)態(tài)生成的實(shí)現(xiàn)類里面委托hanlder去調(diào)用原始實(shí)現(xiàn)類方法,CGLIB則使用的繼承機(jī)制,具體說被代理類和代理類是繼承關(guān)系,所以代理類是可以賦值給被代理類的,如果被代理類有接口,那么代理類也可以賦值給接口。
到此這篇關(guān)于Java實(shí)例講解動(dòng)態(tài)代理的文章就介紹到這了,更多相關(guān)Java動(dòng)態(tài)代理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 一文了解Java動(dòng)態(tài)代理的原理及實(shí)現(xiàn)
- Java實(shí)現(xiàn)JDK動(dòng)態(tài)代理的原理詳解
- 深入理解Java動(dòng)態(tài)代理與靜態(tài)代理
- Java的動(dòng)態(tài)代理和靜態(tài)代理詳解
- Java動(dòng)態(tài)代理的示例詳解
- Java?靜態(tài)代理與動(dòng)態(tài)代理解析
- 帶你重新認(rèn)識(shí)Java動(dòng)態(tài)代理
- Java實(shí)現(xiàn)動(dòng)態(tài)代理的實(shí)例代碼
- Java深入分析動(dòng)態(tài)代理
相關(guān)文章
SpringMVC?Restful風(fēng)格與中文亂碼問題解決方案介紹
Restful就是一個(gè)資源定位及資源操作的風(fēng)格,不是標(biāo)準(zhǔn)也不是協(xié)議,只是一種風(fēng)格,是對(duì)http協(xié)議的詮釋,下面這篇文章主要給大家介紹了關(guān)于SpringMVC對(duì)Restful風(fēng)格支持的相關(guān)資料,需要的朋友可以參考下2022-10-10
SpringBoot 如何實(shí)現(xiàn)自定義Redis序列化
這篇文章主要介紹了SpringBoot 如何實(shí)現(xiàn)自定義Redis序列化方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
idea新建聚合項(xiàng)目并附上標(biāo)簽的詳細(xì)過程
這篇文章主要介紹了idea新建聚合項(xiàng)目并附上標(biāo)簽的詳細(xì)過程,本文通過實(shí)例圖文相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08
重新認(rèn)識(shí)Java中的ThreadLocal
ThreadLocal是JDK包提供的,它提供線程本地變量,如果創(chuàng)建一個(gè)ThreadLocal變量,那么訪問這個(gè)變量的每個(gè)線程都會(huì)有這個(gè)變量的一個(gè)副本,在實(shí)際多線程操作的時(shí)候,操作的是自己本地內(nèi)存中的變量,從而規(guī)避了線程安全問題2021-05-05
springboot如何去獲取前端傳遞的參數(shù)的實(shí)現(xiàn)
這篇文章主要介紹了springboot如何去獲取前端傳遞的參數(shù)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
SpringBoot結(jié)合Neo4j自定義cypherSql的方法
這篇文章主要介紹了SpringBoot結(jié)合Neo4j自定義cypherSql,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-11-11
Java Swing中的表格(JTable)和樹(JTree)組件使用實(shí)例
這篇文章主要介紹了Java Swing中的表格(JTable)和樹(JTree)組件使用實(shí)例,本文同時(shí)講解了表格和樹的基本概念、常用方法、代碼實(shí)例,需要的朋友可以參考下2014-10-10

