Java 動(dòng)態(tài)代理的多種實(shí)現(xiàn)方式
一、動(dòng)態(tài)代理簡(jiǎn)介
優(yōu)勢(shì):在不修改源碼的情況下,對(duì)目標(biāo)方法進(jìn)行相應(yīng)的增強(qiáng)。
作用:完成程序功能之間的松耦合。
二、動(dòng)態(tài)代理的多種實(shí)現(xiàn)
- JDK代理:基于接口的動(dòng)態(tài)代理技術(shù)(缺點(diǎn),目標(biāo)對(duì)象必須有接口,如果沒有接口,則無法完成動(dòng)態(tài)代理的實(shí)現(xiàn))
- cglib代理:基于父類的動(dòng)態(tài)代理技術(shù)
兩者的區(qū)別如圖所示:

1. 基于JDK的實(shí)現(xiàn)
目標(biāo)接口類:
public interface TargetInterface {
public void save();
public void print(String str);
}
目標(biāo)類:
public class Target implements TargetInterface{
public void save() {
System.out.println("save running...");
}
public void print(String str) {
System.out.println(str);
}
}
增強(qiáng)類:
public class Advice {
public void before() {
System.out.println("前置增強(qiáng)");
}
public void after() {
System.out.println("后置增強(qiáng)");
}
}
測(cè)試類:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyTest {
public static void main(String[] args) {
//目標(biāo)對(duì)象
final Target target = new Target();
//增強(qiáng)對(duì)象
final Advice advice = new Advice();
TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance(
target.getClass().getClassLoader(), //目標(biāo)對(duì)象類加載器
target.getClass().getInterfaces(), //目標(biāo)對(duì)象相同的接口字節(jié)碼對(duì)象數(shù)組
new InvocationHandler() {
//調(diào)用代理對(duì)象的任何方法,實(shí)質(zhì)執(zhí)行的都是invoke方法
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
advice.before(); //前置增強(qiáng)
Object invoke = method.invoke(target, args); //執(zhí)行目標(biāo)方法
advice.after(); //后置增強(qiáng)
System.out.println();
return invoke;
}
});
//代理對(duì)象的方法測(cè)試
proxyInstance.save();
proxyInstance.print("JDK動(dòng)態(tài)代理");
}
}
運(yùn)行截圖:

2. 基于cglib的實(shí)現(xiàn)
需要導(dǎo)入Jar包,如果是maven項(xiàng)目,則在pom.xml文件加入如下配置:
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>4.2.4.RELEASE</version> </dependency>
目標(biāo)類:
public class Target {
public void save() {
System.out.println("save running...");
}
public void print(String str) {
System.out.println(str);
}
}
增強(qiáng)類:
public class Advice {
public void before() {
System.out.println("前置增強(qiáng)");
}
public void after() {
System.out.println("后置增強(qiáng)");
}
}
測(cè)試類:
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
public class ProxyTest {
public static void main(String[] args) {
final Target target = new Target();
final Advice advice = new Advice();
//返回值就是動(dòng)態(tài)生成的代理對(duì)象,基于cglib
//創(chuàng)建增強(qiáng)器
Enhancer enhancer = new Enhancer();
//設(shè)置父類(目標(biāo))
enhancer.setSuperclass(Target.class);
//設(shè)置回調(diào)
enhancer.setCallback(new MethodInterceptor() {
public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{
advice.before();
Object invoke = method.invoke(target, obj);
advice.after();
System.out.println();
return invoke;
}
});
//創(chuàng)建代理對(duì)象
Target proxy = (Target)enhancer.create();
//測(cè)試代理方法
proxy.save();
proxy.print("基于cglib實(shí)現(xiàn)動(dòng)態(tài)規(guī)劃");
}
}
運(yùn)行截圖:

三、為什么要有基于cglib的實(shí)現(xiàn)
使用JDK動(dòng)態(tài)代理實(shí)現(xiàn)時(shí),最大限制是被增強(qiáng)對(duì)象必須實(shí)現(xiàn)接口,并且增強(qiáng)的方法只能是接口中聲明的方法。但在實(shí)際的項(xiàng)目中,可能總是存在對(duì)不實(shí)現(xiàn)業(yè)務(wù)接口的對(duì)象進(jìn)行增強(qiáng)的需求,這時(shí)JDK動(dòng)態(tài)代理將無能為力。
四、兩種方式的適用場(chǎng)景
JDK動(dòng)態(tài)代理
優(yōu)點(diǎn)
- 不依賴第三方j(luò)ar包, 使用方便
- 隨著JDK的升級(jí),JDK動(dòng)態(tài)代理的性能在穩(wěn)步提升
缺點(diǎn)
- 只能代理實(shí)現(xiàn)了接口的類
- 執(zhí)行速度較慢
適用場(chǎng)景
- 如果你的程序需要頻繁、反復(fù)地創(chuàng)建代理對(duì)象,則JDK動(dòng)態(tài)代理在性能上更占優(yōu)。
cglib
優(yōu)點(diǎn)
由于是動(dòng)態(tài)生成字節(jié)碼實(shí)現(xiàn)代理,因此代理對(duì)象的執(zhí)行速度較快, 約為JDK動(dòng)態(tài)代理的1.5 ~ 2倍
可以代理沒有實(shí)現(xiàn)接口的對(duì)象
缺點(diǎn)
- 不能代理final類
- 動(dòng)態(tài)生成字節(jié)碼雖然執(zhí)行較快,但是生成速度很慢,根據(jù)網(wǎng)上一些人的測(cè)試結(jié)果,cglib創(chuàng)建代理對(duì)象的速度要比JDK慢10 ~ 15倍。
適用場(chǎng)景
- 不需要頻繁創(chuàng)建代理對(duì)象的應(yīng)用,如Spring中默認(rèn)的單例bean,只需要在容器啟動(dòng)時(shí)生成一次代理對(duì)象。
以上就是Java 動(dòng)態(tài)代理的多種實(shí)現(xiàn)方式的詳細(xì)內(nèi)容,更多關(guān)于Java 動(dòng)態(tài)代理的實(shí)現(xiàn)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
springboot2啟動(dòng)時(shí)執(zhí)行,初始化(或定時(shí)任務(wù))servletContext問題
這篇文章主要介紹了springboot2啟動(dòng)時(shí)執(zhí)行,初始化(或定時(shí)任務(wù))servletContext問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01
SpringCloud負(fù)載均衡spring-cloud-starter-loadbalancer解讀
這篇文章主要介紹了SpringCloud負(fù)載均衡spring-cloud-starter-loadbalancer使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-03-03
Mybatis-Plus根據(jù)時(shí)間段去查詢數(shù)據(jù)的實(shí)現(xiàn)示例
這篇文章主要介紹了Mybatis-Plus根據(jù)時(shí)間段去查詢數(shù)據(jù)的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
Java JDBC連接數(shù)據(jù)庫常見操作總結(jié)
這篇文章主要介紹了Java JDBC連接數(shù)據(jù)庫常見操作,結(jié)合實(shí)例形式總結(jié)分析了java基于jdbc連接mysql、Oracle數(shù)據(jù)庫及連接池相關(guān)操作技巧,需要的朋友可以參考下2019-03-03
Java實(shí)現(xiàn)將byte[]轉(zhuǎn)換為File對(duì)象
這篇文章將通過一個(gè)簡(jiǎn)單的例子為大家演示Java如何實(shí)現(xiàn) byte[] 轉(zhuǎn)換為 File 對(duì)象,并將其上傳到外部服務(wù)器,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-03-03
Java LinkedList的實(shí)現(xiàn)原理圖文詳解
今天小編就為大家分享一篇關(guān)于Java LinkedList的實(shí)現(xiàn)原理圖文詳解,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-01-01
MyBatis中動(dòng)態(tài)sql的實(shí)現(xiàn)方法示例
這篇文章主要給大家介紹了關(guān)于MyBatis中動(dòng)態(tài)sql的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-11-11

