【设计模式】代理模式 ( 动态代理使用流程 | 创建目标对象 | 创建被代理对象 | 创建调用处理程序 | 动态创建代理对象 | 动态代理调用 )
文章目錄
- 前言
- 一、靜態(tài)代理的弊端
- 二、動(dòng)態(tài)代理的優(yōu)勢(shì)
- 三、動(dòng)態(tài)代理使用流程
- 1、目標(biāo)對(duì)象接口
- 2、被代理對(duì)象
- 3、調(diào)用處理程序
- 4、客戶端
- 四、動(dòng)態(tài)生成 代理對(duì)象 類 的 字節(jié)碼 文件數(shù)據(jù)
前言
代理模式結(jié)構(gòu) : 代理模式中的元素有 客戶端 , 主題對(duì)象 , 被代理對(duì)象 , 代理對(duì)象 ;
- 客戶端 持有 主題對(duì)象 , 調(diào)用其方法 ;
- 代理對(duì)象 和 被代理對(duì)象 都是 主題 的子類 ;
- 代理對(duì)象 持有 被代理對(duì)象 , 可以調(diào)用 被代理對(duì)象 的方法 ;
代理模式的核心 : 代理對(duì)象 與 被代理對(duì)象 都實(shí)現(xiàn)同一個(gè)父類或接口 , 這樣在客戶端使用時(shí) , 客戶端 感覺(jué)自己與 被代理對(duì)象 溝通 , 但用戶實(shí)際上與 代理對(duì)象 進(jìn)行溝通 ;
一、靜態(tài)代理的弊端
靜態(tài)代理 中 , 代理對(duì)象 和 被代理對(duì)象 必須實(shí)現(xiàn) 主題對(duì)象 接口 , 如果 主題對(duì)象 接口發(fā)生改變 , 則相應(yīng)的 代理對(duì)象 和 被代理對(duì)象 都要進(jìn)行相應(yīng)修改 ;
二、動(dòng)態(tài)代理的優(yōu)勢(shì)
動(dòng)態(tài)代理 解決了 靜態(tài)代理的上述問(wèn)題 , 不需要手動(dòng)創(chuàng)建代理對(duì)象 , 由 Java 虛擬機(jī)實(shí)現(xiàn) 代理對(duì)象 , 該代理對(duì)象自動(dòng)實(shí)現(xiàn) 主題對(duì)象 的接口 ;
動(dòng)態(tài)代理執(zhí)行時(shí) , 動(dòng)態(tài)地創(chuàng)建了字節(jié)碼文件 , 生成了代理類 ;
三、動(dòng)態(tài)代理使用流程
動(dòng)態(tài)代理使用流程 :
-
① 創(chuàng)建目標(biāo)對(duì)象 : 創(chuàng)建 目標(biāo)對(duì)象 接口 ;
-
② 創(chuàng)建被代理對(duì)象 : 創(chuàng)建 被代理對(duì)象 , 實(shí)現(xiàn) 目標(biāo)對(duì)象 接口 ;
-
③ 創(chuàng)建調(diào)用處理程序 : 創(chuàng)建 InvocationHandler 子類對(duì)象 , 內(nèi)部持有 被代理對(duì)象 , 在 invoke 方法中 , 返回 method.invoke(subject, args) ;
-
④ 動(dòng)態(tài)創(chuàng)建代理對(duì)象 : 調(diào)用 Proxy.newProxyInstance 創(chuàng)建 代理對(duì)象 實(shí)例對(duì)象 , 由 JVM 自動(dòng)創(chuàng)建代理對(duì)象類 , 然后再創(chuàng)建對(duì)應(yīng)的實(shí)例對(duì)象 ;
-
⑤ 動(dòng)態(tài)代理調(diào)用 : 調(diào)用 代理對(duì)象 實(shí)例的相關(guān) 目標(biāo)對(duì)象 接口 方法 ;
1、目標(biāo)對(duì)象接口
/*** 目標(biāo)接口* 代理對(duì)象 和 被代理對(duì)象 都要實(shí)現(xiàn)該接口*/ public interface Subject {void request(); }
2、被代理對(duì)象
/*** 被代理對(duì)象*/ public class RealSubject implements Subject {@Overridepublic void request() {System.out.println("被代理對(duì)象 RealSubject request()");} }
3、調(diào)用處理程序
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method;public class DynamicInvocationHandler implements InvocationHandler {/*** 持有的 被代理對(duì)象*/private Subject subject;public DynamicInvocationHandler(Subject subject) {this.subject = subject;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {// 調(diào)用真實(shí)的 被代理對(duì)象 的方法// 被代理對(duì)象的所有的方法的調(diào)用都會(huì)傳到該方法中進(jìn)行處理Object object = method.invoke(subject, args);return object;} }
4、客戶端
import java.lang.reflect.Proxy;public class Client {public static void main(String[] args) {// 被代理對(duì)象Subject realSubject = new RealSubject();// 創(chuàng)建調(diào)用處理程序 , 內(nèi)部持有被代理對(duì)象DynamicInvocationHandler dynamicInvocationHandler =new DynamicInvocationHandler(realSubject);// 生成動(dòng)態(tài)代理類Subject subject = (Subject) Proxy.newProxyInstance(realSubject.getClass().getClassLoader(),realSubject.getClass().getInterfaces(),dynamicInvocationHandler);// 動(dòng)態(tài)代理調(diào)用subject.request();} }
執(zhí)行結(jié)果 :
四、動(dòng)態(tài)生成 代理對(duì)象 類 的 字節(jié)碼 文件數(shù)據(jù)
動(dòng)態(tài)代理 中的 代理對(duì)象對(duì)應(yīng)的 字節(jié)碼類 是由 Java 虛擬機(jī)自動(dòng)生成的 , 在 java.lang.reflect.Proxy 中 , 調(diào)用 ProxyGenerator.generateProxyClass 方法 , 生成了 代理對(duì)象 類 , 返回的 byte[] 數(shù)據(jù)就是字節(jié)碼類對(duì)應(yīng)的二進(jìn)制數(shù)據(jù) ;
/** Generate the specified proxy class.*/byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces, accessFlags);ProxyGenerator 的源碼需要下載 JDK 源碼查看 , 在 IntelliJ IDEA 開(kāi)發(fā)環(huán)境中無(wú)法查看 ;
網(wǎng)上找到了一篇博客 , 對(duì)此描述的很清楚 JDK動(dòng)態(tài)代理[4]----ProxyGenerator生成代理類的字節(jié)碼文件解析 ;
ProxyGenerator 中的 generateProxyClass 方法中 , 主要調(diào)用了 generateClassFile 方法 , 按照 Class 字節(jié)碼的規(guī)范 , 按照順序依次寫入 魔數(shù) , 次版本號(hào) , 主版本號(hào) , 常量池 , 訪問(wèn)修飾符 , 類索引 等數(shù)據(jù) ;
總結(jié)
以上是生活随笔為你收集整理的【设计模式】代理模式 ( 动态代理使用流程 | 创建目标对象 | 创建被代理对象 | 创建调用处理程序 | 动态创建代理对象 | 动态代理调用 )的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 【设计模式】面向对象 - 多态 ( 面向
- 下一篇: asp.net ajax控件工具集 Au