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ú)寫(xiě)一個(gè)代理類)。 實(shí)際應(yīng)用場(chǎng)景非常非常少,日常開(kāi)發(fā)幾乎看不到使用靜態(tài)代理的場(chǎng)景。
角色分析:
- 抽象角色:一般會(huì)使用接口或者抽象類來(lái)解決
- 真實(shí)角色:被代理的角色
- 代理角色:代理真實(shí)角色,代理真實(shí)角色后,我們一般會(huì)做一些附屬操作
- 客戶:訪問(wèn)代理對(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、客服端訪問(wèn)代理角色
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角度來(lái)看,靜態(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)生成的,不是我們直接寫(xiě)好的!
- 動(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 角度來(lái)說(shuō),動(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 來(lái)自定義處理邏輯。 當(dāng)我們的動(dòng)態(tài)代理對(duì)象調(diào)用一個(gè)方法時(shí),這個(gè)方法的調(diào)用就會(huì)被轉(zhuǎn)發(fā)到實(shí)現(xiàn)InvocationHandler 接口類的 invoke 方法來(lái)調(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() :主要通過(guò)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é)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
使用jmx?exporter采集kafka指標(biāo)示例詳解
這篇文章主要為大家介紹了使用jmx?exporter采集kafka指標(biāo)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
使用ObjectMapper把Json轉(zhuǎn)換為復(fù)雜的實(shí)體類
這篇文章主要介紹了使用ObjectMapper把Json轉(zhuǎn)換為復(fù)雜的實(shí)體類操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08
java+mysql實(shí)現(xiàn)登錄和注冊(cè)功能
這篇文章主要為大家詳細(xì)介紹了java+mysql實(shí)現(xiàn)登錄和注冊(cè)功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
Json 自定義使用函數(shù)的簡(jiǎn)單實(shí)例
下面小編就為大家?guī)?lái)一篇Json 自定義使用函數(shù)的簡(jiǎn)單實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-10-10
AOP之事務(wù)管理<aop:advisor>的兩種配置方式
這篇文章主要介紹了AOP之事務(wù)管理<aop:advisor>的兩種配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
如何寫(xiě)好一個(gè)Spring組件的實(shí)現(xiàn)步驟
這篇文章主要介紹了如何寫(xiě)好一個(gè)Spring組件的實(shí)現(xiàn)步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06

