Java兩種方式實(shí)現(xiàn)動(dòng)態(tài)代理
一、JDK動(dòng)態(tài)代理
Java 在 java.lang.reflect 包中有自己的代理支持,該類(Proxy.java)用于動(dòng)態(tài)生成代理類,只需傳入目標(biāo)接口、目標(biāo)接口的類加載器以及 InvocationHandler 便可為目標(biāo)接口生成代理類及代理對(duì)象。我們稱這個(gè)Java技術(shù)為:動(dòng)態(tài)代理
@CallerSensitive
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
throws IllegalArgumentException
{
//...
}
在 Java 中規(guī)定,要想產(chǎn)生一個(gè)對(duì)象的代理對(duì)象,那么這個(gè)對(duì)象必須要有一個(gè)接口,因此 interfaces 必須是一個(gè)接口。
在動(dòng)態(tài)代理技術(shù)里,由于不管用戶調(diào)用代理對(duì)象的什么方法,都是調(diào)用開發(fā)人員編寫的 InvocationHandler 的 invoke 方法(這相當(dāng)于 invoke 方法攔截到了代理對(duì)象的方法調(diào)用)。
因此 JDK 動(dòng)態(tài)代理的整體流程為:
- 實(shí)現(xiàn) InvocationHandler,用來處理對(duì)象攔截后的邏輯。(該對(duì)象必須是接口,或者父類是接口)
- 使用 Proxy.newProxyInstance 產(chǎn)生代理對(duì)象。
以下是一個(gè)用 JDK 動(dòng)態(tài)代碼實(shí)現(xiàn) AOP 的具體例子:
1.目標(biāo)(Target)類
public interface UserService {
void eat();
}
public class UserServiceImpl implements UserService {
@Override
public void eat() {
System.out.println("吃東西");
}
}
2.切面(Aspect)類
public class MyAspect {
/**
* 前置通知
*/
public void before() {
System.out.print("先洗手再");
}
}
3. 織入(Weaving)過程
/**
* 產(chǎn)生代理對(duì)象的工廠類
*/
public class MyFactoryBean {
private MyFactoryBean() {
}
public static UserService getInstance() {
// target : 目標(biāo)類
final UserService userService = new UserServiceImpl();
// Aspect : 切面類
final MyAspect myAspect = new MyAspect();
// Weaving : 織入,也就是產(chǎn)生代理的過程
UserService proxyInstance = (UserService) Proxy.newProxyInstance(UserService.class.getClassLoader(),
new Class[]{UserService.class}, (Object proxy, Method method, Object[] args) -> {
// 模擬切點(diǎn) - pointcut
if ("eat".equals(method.getName())) {
myAspect.before();
}
return method.invoke(userService, args);
});
return proxyInstance;
}
}
public static void main(String[] args) {
UserService userService = MyFactoryBean.getInstance();
// 先洗手再吃東西
userService.eat();
}
想想看,這其實(shí)跟我們平常使用的 AOP 已經(jīng)很相似了,Spring 里面定義了前置通知(@Before)、異常通知(@AfterThrowing)等等,Spring 只是換成了甄別這些注解來選擇什么時(shí)候調(diào)用通知方法,另外,Spring 還通過切點(diǎn)表達(dá)式來選擇目標(biāo)類和切入點(diǎn)。
二、CGLIB動(dòng)態(tài)代理
CGLIB 動(dòng)態(tài)代理需要引入第三方的庫,它通過修改代理對(duì)象生成子類的方式來實(shí)現(xiàn)調(diào)用攔截,代理對(duì)象不需要實(shí)現(xiàn)接口,但是代理類不能是 final,代理的方法也不能是 final。
/**
* 產(chǎn)生代理對(duì)象的工廠類
*/
public class MyFactoryBean {
private MyFactoryBean() {
}
public static UserService getInstance() {
// target : 目標(biāo)類
final UserService userService = new UserServiceImpl();
// Aspect : 切面類
final MyAspect myAspect = new MyAspect();
// Weaving : 織入,也就是產(chǎn)生代理的過程
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(userService.getClass());
enhancer.setUseCache(false);
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
// 模擬 pointcut-切點(diǎn)
if ("eat".equals(method.getName())) {
myAspect.before();
}
return methodProxy.invokeSuper(o, objects);
}
});
return (UserService) enhancer.create();
}
public static void main(String[] args) {
UserService proxyInstance = MyFactoryBean.getInstance();
// 先洗手再吃東西
proxyInstance.eat();
}
}
三、總結(jié)
在 JDK 中實(shí)現(xiàn)動(dòng)態(tài)代理時(shí),要求代理類必須是接口或繼承接口的類,因?yàn)?JDK 最后生成的 proxy class 其實(shí)就是實(shí)現(xiàn)了代理類所代理的接口并且繼承了 java 中的 Proxy 類(繼承 Proxy 類是為了判斷該類是否為代理類),通過反射找到接口的方法,調(diào)用 InvocationHandler的invoke 方法實(shí)現(xiàn)攔截。
CGLIB 字節(jié)碼增強(qiáng)是JDK動(dòng)態(tài)代理的一個(gè)很好的補(bǔ)充, CGLIB 中最后生成的 proxy class 是一個(gè)繼承代理類所代理的 class,通過重寫被代理類中的非 final 的方法實(shí)現(xiàn)代理。
總結(jié)為:
- JDK 動(dòng)態(tài)代理:代理類必須是接口或繼承接口的類。
- CGLIB 字節(jié)碼增強(qiáng): 代理類不能是 final,代理的方法也不能是 final(繼承限制) 。
關(guān)于在 Spring 的 AOP 中采用何種代理手段,我們不強(qiáng)加限制的話,會(huì)根據(jù)類是否有接口來區(qū)別對(duì)待:
- 當(dāng)一個(gè)類有接口的時(shí)候,就會(huì)選用 JDK 的動(dòng)態(tài)代理。
- 當(dāng)一個(gè)類沒有實(shí)現(xiàn)接口的時(shí)候,就會(huì)選用 CGLIB 代理的方式。
以上就是Java兩種方式實(shí)現(xiàn)動(dòng)態(tài)代理的詳細(xì)內(nèi)容,更多關(guān)于Java 動(dòng)態(tài)代理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Java JDK動(dòng)態(tài)代理(AOP)用法及實(shí)現(xiàn)原理詳解
- 詳解Java JDK動(dòng)態(tài)代理
- 詳解Java Cglib動(dòng)態(tài)代理
- Java簡(jiǎn)單實(shí)現(xiàn)動(dòng)態(tài)代理模式過程解析
- Java JDK動(dòng)態(tài)代理實(shí)現(xiàn)原理實(shí)例解析
- 詳細(xì)分析java 動(dòng)態(tài)代理
- Java動(dòng)態(tài)代理語法Proxy類原理詳解
- Java動(dòng)態(tài)代理靜態(tài)代理實(shí)例分析
- Java代理模式實(shí)例詳解【靜態(tài)代理與動(dòng)態(tài)代理】
- JAVA使用動(dòng)態(tài)代理對(duì)象進(jìn)行敏感字過濾代碼實(shí)例
- Java動(dòng)態(tài)代理模式的深入揭秘
- Java 動(dòng)態(tài)代理的多種實(shí)現(xiàn)方式
相關(guān)文章
一文吃透Spring?Cloud?gateway自定義錯(cuò)誤處理Handler
這篇文章主要為大家介紹了一文吃透Spring?Cloud?gateway自定義錯(cuò)誤處理Handler方法,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03
Mybatis-Plus開發(fā)提速器mybatis-plus-generator-ui詳解
這篇文章主要介紹了Mybatis-Plus開發(fā)提速器mybatis-plus-generator-ui,本文簡(jiǎn)要介紹一款基于Mybatis-Plus的代碼自助生成器,文章通過實(shí)例集成的方式來詳細(xì)講解mybatis-plus-generator-ui,從相關(guān)概念到實(shí)際集成案例,以及具體的擴(kuò)展開發(fā)介紹,需要的朋友可以參考下2022-11-11
SpringBoot?AOP?Redis實(shí)現(xiàn)延時(shí)雙刪功能實(shí)戰(zhàn)
本文主要介紹了SpringBoot?AOP?Redis實(shí)現(xiàn)延時(shí)雙刪功能實(shí)戰(zhàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
基于MybatisPlus插件TenantLineInnerInterceptor實(shí)現(xiàn)多租戶功能
這篇文章主要介紹了基于MybatisPlus插件TenantLineInnerInterceptor實(shí)現(xiàn)多租戶功能,需要的朋友可以參考下2021-11-11
Java 獲取當(dāng)前類名和方法名的實(shí)現(xiàn)方法
這篇文章主要介紹了 Java 獲取當(dāng)前類名和方法名的實(shí)現(xiàn)方法的相關(guān)資料,這里不僅提供了實(shí)現(xiàn)方法并比較幾種方法的效率,需要的朋友可以參考下2017-07-07

