Java的動(dòng)態(tài)代理和靜態(tài)代理詳解
0、代理模式
為什么要學(xué)習(xí)代理模式?這是SpringAOP的底層【SpringAOP和SpringMVC】
代理模式的分類:
- 靜態(tài)代理
- 動(dòng)態(tài)代理
1、靜態(tài)代理
靜態(tài)代理中,我們對(duì)目標(biāo)對(duì)象的每個(gè)方法的增強(qiáng)都是手動(dòng)完成的(后面會(huì)具體演示代碼_),非常不靈活(比如接口一旦新增加方法,目標(biāo)對(duì)象和代理對(duì)象都要進(jìn)行修改)且麻煩(_需要對(duì)每個(gè)目標(biāo)類都單獨(dú)寫一個(gè)代理類)。 實(shí)際應(yīng)用場景非常非常少,日常開發(fā)幾乎看不到使用靜態(tài)代理的場景。
角色分析:
- 抽象角色:一般會(huì)使用接口或者抽象類來解決
- 真實(shí)角色:被代理的角色
- 代理角色:代理真實(shí)角色,代理真實(shí)角色后,我們一般會(huì)做一些附屬操作
- 客戶:訪問代理對(duì)象的人!
代碼步驟:
1、接口
public interface Rent {
public void rent();
}
2、真實(shí)角色
//房東
public class Host implements Rent {
public void rent() {
System.out.println("房東要租房子");
}
}
3、代理角色
public class Proxy implements Rent{
private Host host;
public Proxy() {
}
public Proxy(Host host) {
this.host = host;
}
public void rent(){
seeHouse();
host.rent();
fare();
}
//看房
public void seeHouse(){
System.out.println("中介帶你看房");
}
//收中介費(fèi)
public void fare(){
System.out.println("中介收費(fèi)");
}
}
4、客服端訪問代理角色
public class Client {
public static void main(String[] args) {
Host host = new Host();
//代理,代理角色一般會(huì)有附屬操作!
Proxy proxy = new Proxy(host);
proxy.rent();
}
}
代理模式的好處:
- 可以使真實(shí)角色的操作更加純粹!不用去關(guān)注一些公共的業(yè)務(wù)
- 公共也就交給代理角色!實(shí)現(xiàn)業(yè)務(wù)的分工!
- 公共業(yè)務(wù)發(fā)生擴(kuò)展的時(shí)候,方便集中管理!
缺點(diǎn):
一個(gè)真實(shí)角色會(huì)產(chǎn)生一個(gè)代理角色;從JVM角度來看,靜態(tài)代理在編譯時(shí)就將接口、實(shí)現(xiàn)類、代理類這些都變成了一個(gè)個(gè)實(shí)際的class文件。
2、 加深理解
AOP,的底層代理模式

3、動(dòng)態(tài)代理
- 動(dòng)態(tài)代理和靜態(tài)代理角色一樣
- 動(dòng)態(tài)代理的代理類是動(dòng)態(tài)生成的,不是我們直接寫好的!
- 動(dòng)態(tài)代理分為兩大類:基于接口的動(dòng)態(tài)代理,基于類的動(dòng)態(tài)代理
- 基于接口——JDK動(dòng)態(tài)代理
- 基于類:cglib動(dòng)態(tài)代理
- java字節(jié)碼實(shí)現(xiàn):javasist
需要了解兩個(gè)類:Proxy: 代理類,InvocationHandler : 調(diào)用處理程序
從 JVM 角度來說,動(dòng)態(tài)代理是在運(yùn)行時(shí)動(dòng)態(tài)生成類字節(jié)碼,并加載到 JVM 中的。
//Proxy是生成動(dòng)態(tài)代理類,提供了創(chuàng)建動(dòng)態(tài)代理類和實(shí)例的靜態(tài)方法,它也是由這些方法創(chuàng)建的所有動(dòng)態(tài)代理類的超類。 //InvocationHandler-- invoke 調(diào)用處理程序并返回接口, 是由代理實(shí)例的調(diào)用處理程序?qū)崿F(xiàn)的接口 。
動(dòng)態(tài)代理的好處:
- 可以使真實(shí)角色的操作更加純粹!不用去關(guān)系一些公共的業(yè)務(wù)
- 公共也就交給代理角色!實(shí)現(xiàn)
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h){
}
1.loader :類加載器,用于加載代理對(duì)象。
2.interfaces : 被代理類實(shí)現(xiàn)的一些接口;
3.h : 實(shí)現(xiàn)了 InvocationHandler 接口的對(duì)象;
要實(shí)現(xiàn)動(dòng)態(tài)代理的話,還必須需要實(shí)現(xiàn)InvocationHandler 來自定義處理邏輯。 當(dāng)我們的動(dòng)態(tài)代理對(duì)象調(diào)用一個(gè)方法時(shí),這個(gè)方法的調(diào)用就會(huì)被轉(zhuǎn)發(fā)到實(shí)現(xiàn)InvocationHandler 接口類的 invoke 方法來調(diào)用。
public interface InvocationHandler {
Object invoke(Object proxy, Method method, Object[] args) throws Throwable;
}
1.proxy :動(dòng)態(tài)生成的代理類
2.method : 與代理類對(duì)象調(diào)用的方法相對(duì)應(yīng)
3.args : 當(dāng)前 method 方法的參數(shù)
動(dòng)態(tài)代理的例子
1、定義接口
public interface Rent {
public void rent();
}
2、實(shí)現(xiàn)租房的接口
public class Host implements Rent {
@Override
public void rent() {
System.out.println("房東要租房");
}
}
3、定義一個(gè)JDK動(dòng)態(tài)代理類
public class DebugInvocationHandler implements InvocationHandler {
/**
* 代理類中的真實(shí)對(duì)象
*/
private final Object target;
public DebugInvocationHandler(Object target){
this.target = target;
}
/**
* 當(dāng)你使用代理對(duì)象調(diào)用方法的時(shí)候?qū)嶋H會(huì)調(diào)用到這個(gè)方法
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//調(diào)用方法前
System.out.println("before method" + method.getName());
Object res = method.invoke(target, args);
//調(diào)用方法后
System.out.println("after method" + method.getName());
return res;
}
}
invoke() 方法: 當(dāng)我們的動(dòng)態(tài)代理對(duì)象調(diào)用原生方法的時(shí)候,最終實(shí)際上調(diào)用到的是 invoke() 方法,然后 invoke() 方法代替我們?nèi)フ{(diào)用了被代理對(duì)象的原生方法。
4、獲取代理對(duì)象的工廠類
public class JdkProxyFactory {
public static Object getProxy(Object target){
return Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new DebugInvocationHandler(target)
);
}
}
getProxy() :主要通過Proxy.newProxyInstance()方法獲取某個(gè)類的代理對(duì)象
5、實(shí)際使用
public static void main(String[] args) {
//Rent rent = new Host();
//Rent rentProxy= (Rent) Proxy.newProxyInstance(rent.getClass().getClassLoader(), rent.getClass().getInterfaces(),new DebugInvocationHandler(rent));
Rent rentProxy = (Rent)JdkProxyFactory.getProxy(new Host());
rentProxy.rent();
}
運(yùn)行上述代理的輸出
before methodrent
房東要租房
after methodrent
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Java編程實(shí)現(xiàn)軌跡壓縮算法開放窗口實(shí)例代碼
這篇文章主要介紹了Java編程實(shí)現(xiàn)軌跡壓縮算法開放窗口實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-11-11
SpringBoot之@ConditionalOnProperty注解使用方法
在平時(shí)業(yè)務(wù)中,我們需要在配置文件中配置某個(gè)屬性來決定是否需要將某些類進(jìn)行注入,讓Spring進(jìn)行管理,而@ConditionalOnProperty能夠?qū)崿F(xiàn)該功能,文中有詳細(xì)的代碼示例,需要的朋友可以參考下2023-05-05
SpringBoot文件上傳同時(shí)接收復(fù)雜參數(shù)的過程詳解
這篇文章主要介紹了SpringBoot文件上傳同時(shí),接收復(fù)雜參數(shù),本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12
SpringSecurity實(shí)現(xiàn)權(quán)限認(rèn)證與授權(quán)的使用示例
本文主要介紹了SpringSecurity實(shí)現(xiàn)權(quán)限認(rèn)證與授權(quán)的使用示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11
Java如何使用elasticsearch進(jìn)行模糊查詢
這篇文章主要介紹了Java如何使用elasticsearch進(jìn)行模糊查詢,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02

