帶你重新認(rèn)識Java動態(tài)代理
什么是動態(tài)代理?
動態(tài)代理就是,在程序運(yùn)行期,創(chuàng)建目標(biāo)對象的代理對象,并對目標(biāo)對象中的方法進(jìn)行功能性增強(qiáng)的一種技術(shù)。在生成代理對象的過程中,目標(biāo)對象不變,代理對象中的方法是目標(biāo)對象方法的增強(qiáng)方法。可以理解為運(yùn)行期間,對象中方法的動態(tài)攔截,在攔截方法的前后執(zhí)行功能操作(也可以對原方法的參數(shù)進(jìn)行操作)。
代理類在程序運(yùn)行期間,創(chuàng)建的代理對象稱之為動態(tài)代理對象。這種情況下,創(chuàng)建的代理對象,并不是事先在Java代碼中定義好的。而是在運(yùn)行期間,根據(jù)我們在動態(tài)代理對象中的“指示”,動態(tài)生成的。也就是說,你想獲取哪個對象的代理,動態(tài)代理就會為你動態(tài)的生成這個對象的代理對象。動態(tài)代理可以對被代理對象的方法進(jìn)行功能增強(qiáng)。有了動態(tài)代理的技術(shù),那么就可以在不修改方法源碼的情況下,增強(qiáng)被代理對象的方法的功能,在方法執(zhí)行前后做任何你想做的事情。
特點:字節(jié)碼隨用隨創(chuàng)建,隨用隨加載
作用:不修改源碼的基礎(chǔ)上對方法增強(qiáng)
正常類創(chuàng)建對象的過程:

動態(tài)代理創(chuàng)建代理對象的過程:

動態(tài)代理的常用兩種方式:
1.基于接口的動態(tài)代理
提供者:JDK
使用JDK官方的Proxy類創(chuàng)建代理對象
注意:代理的目標(biāo)對象必須實現(xiàn)接口(至少一個)
2.基于類的動態(tài)代理
提供者:第三方 CGLib
使用CGLib的Enhancer類創(chuàng)建代理對象
注意:被代理類不能用 final 修飾的類(最終類)。如果報 asmxxxx 異常,需要導(dǎo)入 asm.jar包
//JDK動態(tài)代理(基于接口的動態(tài)代理) Proxy.newProxyInstance(三個參數(shù)); ClassLoader:類加載器 它是用于加載代理對象字節(jié)碼的。和被代理對象使用相同的類加載器。(固定寫法) Class[]:字節(jié)碼數(shù)組 它是用于讓代理對象和被代理對象有相同方法。(固定寫法) InvocationHandler:用于提供增強(qiáng)的代碼 它是讓我們寫如何代理。我們一般都是寫一個該接口的實現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須 InvocationHandler該接口的實現(xiàn)類是誰用誰寫,此時我們用就需要我們自己寫
此處以一個演員的例子為例:
在很久以前,演員和劇組都是直接見面聯(lián)系的。沒有中間人環(huán)節(jié)。
而隨著時間的推移,產(chǎn)生了一個新興職業(yè):經(jīng)紀(jì)人(中間人),這個時候劇組再想找演員就需要通過經(jīng)紀(jì)人來找了。下面我們就用代碼演示出來。
package com.haust.service;
public interface IActor {
/**
* 基本演出
* @param money
*/
public void basicAct(float money);
/**
* 危險演出
* @param money
*/
public void dangerAct(float money);
}
package com.haust.serviceImpl;
import com.haust.service.IActor;
public class Actor implements IActor {
/**
* 一個演員
*/
//實現(xiàn)了接口,就表示具有接口中的方法實現(xiàn)。即:符合經(jīng)紀(jì)公司的要求
@Override
public void basicAct(float money) {
System.out.println("拿到錢,開始基本的表演:"+money);
}
@Override
public void dangerAct(float money) {
System.out.println("拿到錢,開始危險的表演:"+money);
}
}
package com.haust.test;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import com.haust.service.IActor;
import com.haust.serviceImpl.Actor;
public class Client {
public static void main(String[] args) {
//一個劇組找演員:
final Actor actor = new Actor();//被代理的類
/**
* 代理:
* 間接。
* 獲取代理對象:
* 要求:
* 被代理類最少實現(xiàn)一個接口
* 創(chuàng)建的方式
* Proxy.newProxyInstance(三個參數(shù))
* 參數(shù)含義:
* ClassLoader:和被代理對象使用相同的類加載器。
* Interfaces:和被代理對象具有相同的行為。實現(xiàn)相同的接口。
* InvocationHandler:如何代理。
*
*/
//(IActor)Proxy.newProxyInstance,這里強(qiáng)制轉(zhuǎn)換必須是接口類型
IActor proxyActor = (IActor)Proxy.newProxyInstance(actor.getClass().getClassLoader(),
actor.getClass().getInterfaces(),
new InvocationHandler() {
/**
* 執(zhí)行被代理對象的任何方法,都會經(jīng)過該方法。
* 此方法有攔截的功能。
*
* 參數(shù):
* proxy:代理對象的引用。不一定每次都用得到
* method:當(dāng)前執(zhí)行的方法對象
* args:執(zhí)行方法所需的參數(shù)
* 返回值:
* 當(dāng)前執(zhí)行被代理對象方法的返回值
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
String name = method.getName();
Float money = (Float) args[0];//執(zhí)行的方法只有一個參數(shù)
Object rtValue = null;
//每個經(jīng)紀(jì)公司對不同演出收費(fèi)不一樣,此處開始判斷
if("basicAct".equals(name)){
//基本演出,沒有 2000 不演
if(money > 2000){
//看上去劇組是給了 8000,實際到演員手里只有 4000
//這就是我們沒有修改原來 basicAct 方法源碼,對方法進(jìn)行了增強(qiáng)
rtValue = method.invoke(actor, money/2);
} }
if("dangerAct".equals(name)){
//危險演出,沒有 5000 不演
if(money > 5000){
//看上去劇組是給了 50000,實際到演員手里只有 25000
//這就是我們沒有修改原來 dangerAct 方法源碼,對方法進(jìn)行了增強(qiáng)
rtValue = method.invoke(actor, money/2);
} }
return rtValue;
}
});
//沒有經(jīng)紀(jì)公司的時候,直接找演員。
// actor.basicAct(1000f);
// actor.dangerAct(5000f);
//劇組無法直接聯(lián)系演員,而是由經(jīng)紀(jì)公司找的演員
proxyActor.basicAct(2000f);//價格低于2000不演
proxyActor.dangerAct(50000f);
}
}
總結(jié):
首先需要創(chuàng)建一個interface然后一個class實現(xiàn)這個interface,然后對這個class進(jìn)行代理,這個class必須實現(xiàn)至少一個接口
基于子類的動態(tài)代理
設(shè)計的類:Enhancer 提供者:第三方cglib庫 如何創(chuàng)建代理對象: 使用Enhancer類中的create方法 創(chuàng)建代理對象的要求: 被代理對象不是最終類(最終類沒有子類) create方法的參數(shù): Class方法的參數(shù): Class:字節(jié)碼 它是用于指定被代理對象的字節(jié)碼 callback:用于提供增強(qiáng)的代碼 它是讓我們寫如何代理。我們一般都是些一個該接口的實現(xiàn)類,通常情況下都是匿名內(nèi)部類,但不是必須的。此接口的實現(xiàn)類都是誰用誰寫。 我們一般寫的都是該接口的子接口實現(xiàn)類:MethodInterceptor //CGLib動態(tài)代理(基于子類的動態(tài)代理) Enhancer.create(兩個參數(shù));
代碼如下:
package com.haust.serviceImpl;
public class Actor{//沒有實現(xiàn)任何接口
/**
* 一個演員
*/
public void basicAct(float money) {
System.out.println("拿到錢,開始基本的表演:"+money);
}
public void dangerAct(float money) {
System.out.println("拿到錢,開始危險的表演:"+money);
}
}
package com.haust.test;
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import com.haust.serviceImpl.Actor;
public class test {
public static void main(String[] args) {
Actor actor = new Actor();//需要創(chuàng)建此被代理的對象
/**
* 基于子類的動態(tài)代理
* 要求:
* 被代理對象不能是最終類
* 用到的類:
* Enhancer
* 用到的方法:
* create(Class, Callback)
* 方法的參數(shù):
* Class:被代理對象的字節(jié)碼
* Callback:如何代理
* @param args
*/
//此時強(qiáng)轉(zhuǎn)的類的類型就是被代理類的類型
Actor cglibActor = (Actor)Enhancer.create(actor.getClass(),new MethodInterceptor() {
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
/**
* 執(zhí)行被代理對象的任何方法,都會經(jīng)過該方法。在此方法內(nèi)部就可以對被代理對象的任何
方法進(jìn)行增強(qiáng)。
*
* 參數(shù):
* 前三個和基于接口的動態(tài)代理是一樣的。
* MethodProxy:當(dāng)前執(zhí)行方法的代理對象。
* 返回值:
* 當(dāng)前執(zhí)行方法的返回值
*/
String name = method.getName();
Float money = (Float) args[0];
Object rtValue = null;
if("basicAct".equals(name)){
//基本演出
if(money > 2000){
rtValue = method.invoke(actor, money/2);
} }
if("dangerAct".equals(name)){
//危險演出
if(money > 5000){
rtValue = method.invoke(actor, money/2);
}
}
return rtValue;
}
});
cglibActor.basicAct(10000);
cglibActor.dangerAct(100000);
}
}
總結(jié):
無論哪種代理方式,都需要創(chuàng)建一個被代理的類(實例)。
不管是基于接口的代理,還是基于子類的代理,均攔截被代理對象的所有方法,然后我們可以對這些方法進(jìn)行增強(qiáng)或者其他一些操作。
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
springboot實現(xiàn)請求參數(shù)驗證的多種方法
在日常開發(fā)中,我們少不了需要對前端的請求參數(shù)的驗證,Spring提供了多種方法來實現(xiàn)請求參數(shù)的驗證,文中通過代碼示例給大家講解的非常詳細(xì),我們一起了解一下吧2023-11-11
java并發(fā)學(xué)習(xí)-CountDownLatch實現(xiàn)原理全面講解
這篇文章主要介紹了java并發(fā)學(xué)習(xí)-CountDownLatch實現(xiàn)原理全面講解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02
詳解mybatis批量插入10萬條數(shù)據(jù)的優(yōu)化過程
這篇文章主要介紹了詳解mybatis批量插入10萬條數(shù)據(jù)的優(yōu)化過程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04

