溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

什么是靜態(tài)代理設(shè)計模式

發(fā)布時間:2021-10-13 14:26:37 來源:億速云 閱讀:161 作者:iii 欄目:編程語言

本篇內(nèi)容主要講解“什么是靜態(tài)代理設(shè)計模式”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學(xué)習(xí)“什么是靜態(tài)代理設(shè)計模式”吧!

靜態(tài)代理設(shè)計模式

代理設(shè)計模式是在程序開發(fā)中使用最多的設(shè)計模式,代理設(shè)計模式的核心是有真實業(yè)務(wù)實現(xiàn)類和代理業(yè)務(wù)實現(xiàn)類,并且代理類要完成比真實業(yè)務(wù)更多的處理操作。

傳統(tǒng)代理設(shè)計模式的弊端

所有的代理設(shè)計模式如果按照設(shè)計要求來說,必須是基于接口的設(shè)計,也就是說需要首先定義出核心接口的組成。
范例:模擬一個消息發(fā)送的代理操作結(jié)構(gòu)(傳統(tǒng)代理設(shè)計)

public class JavaAPIDemo {public static void main(String[] args)throws Exception{
        IMessage message=new MessageProxy(new MessageReal());
        message.send();
    }
}interface IMessage{   //傳統(tǒng)代理設(shè)計必須有接口public void send();   //業(yè)務(wù)方法}class MessageReal implements IMessage {@Overridepublic void send() {
        System.out.println("【發(fā)送消息】www.mldn.cn");
    }
}class MessageProxy implements IMessage {   //代理類private IMessage message;  //代理對象,一定是業(yè)務(wù)接口實例public MessageProxy(IMessage message){this.message=message;
    }@Overridepublic void send() {if(this.connect()){this.message.send();  //消息的發(fā)送處理this.close();
        }
    }public boolean connect(){
        System.out.println("【消息代理】進行消息發(fā)送通道的連接。");return true;
    }public void close(){
        System.out.println("【消息代理】關(guān)閉消息通道。");
    }
}

執(zhí)行結(jié)果:
什么是靜態(tài)代理設(shè)計模式

以上的操作代碼是一個最為標(biāo)準(zhǔn)的代理設(shè)計,但是如果要進一步的去思考會發(fā)現(xiàn)客戶端的接口與具體的子類產(chǎn)生了耦合問題,所以這樣的操作如果從實際的開發(fā)來講,最好再引入工廠設(shè)計模式進行代理對象的獲取。

以上的代理設(shè)計模式為靜態(tài)代理設(shè)計,這種靜態(tài)代理涉及的特點在于:一個代理類只為一個接口服務(wù),如果現(xiàn)在準(zhǔn)備有3000個業(yè)務(wù)接口,則按照此種做法就意味著需要編寫3000個代理類,并且這些代理類操作形式類似。

所以現(xiàn)在需要解決的問題在于:如何可以讓一個代理類滿足于所有的業(yè)務(wù)接口操作要求。

動態(tài)代理設(shè)計模式

通過靜態(tài)代理設(shè)計模式的缺陷可以發(fā)現(xiàn),最好的做法是為所有功能一致的業(yè)務(wù)操作接口提供有統(tǒng)一的代理處理操作,而這就可以通過動態(tài)代理機制來實現(xiàn),但是在動態(tài)代理機制中需要考慮到如下幾點問題:

  • 不管是動態(tài)代理類還是靜態(tài)代理類都一定要接收真實業(yè)務(wù)實現(xiàn)子類對象;

  • 由于動態(tài)代理類不再與某一個具體的接口進行捆綁,所以應(yīng)該可以動態(tài)獲取類的接口信息。

什么是靜態(tài)代理設(shè)計模式
動態(tài)代理設(shè)計模式

在進行動態(tài)代理實現(xiàn)的操作中,首先需要關(guān)注的就是一個InvocationHandler接口,這個接口規(guī)定了代理方法的執(zhí)行。

public interface InvocationHandler{/**     * 代理方法調(diào)用,代理主體類中執(zhí)行的方法最終都是此方法     * @param proxy 要代理的對象     * @param method 要執(zhí)行的接口方法名稱     * @param args 傳遞的參數(shù)     * @return 某一個方法的返回值     * @throws Throwable 方法調(diào)用時出現(xiàn)的錯誤繼續(xù)向上拋出     */public Object invoke(Object proxy, Method method, Object[] args) throws Throwable;
}

在進行動態(tài)代理設(shè)計時,對于動態(tài)對象的創(chuàng)建是由JVM底層完成的,此時主要依靠的是java.lang.reflect.Proxy程序類,而這個類中只提供了一個核心方法:
代理對象:public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

  • ClassLoader loader:獲取當(dāng)前真實主體類的ClassLoader;

  • Class<?>[] interfaces:代理是圍繞接口進行的,所以一定要獲取真實主體類的接口信息;

  • InvocationHandler h:代理處理的方法;

范例:實現(xiàn)動態(tài)代理機制

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;public class JavaAPIDemo {public static void main(String[] args)throws Exception{
        IMessage msg=(IMessage)new MLDNProxy().bind(new MessageReal());
        msg.send();
    }
}class MLDNProxy implements InvocationHandler{private Object target;  //保存真實業(yè)務(wù)對象/**
     * 進行真實業(yè)務(wù)對象與代理業(yè)務(wù)對象之間的綁定處理
     * @param target 真實業(yè)務(wù)對象
     * @return Proxy生成的代理業(yè)務(wù)對象
     */public Object bind(Object target){this.target = target;return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),this);
    }public boolean connect(){
        System.out.println("【消息代理】進行消息發(fā)送通道的連接。");return true;
    }public void close(){
        System.out.println("【消息代理】關(guān)閉消息通道。");
    }@Overridepublic Object invoke(Object pro, Method method, Object[] args) throws Throwable {
        System.out.println("*****【執(zhí)行方法: 】"+method);
        Object returnData = null;if(this.connect()){
            returnData = method.invoke(this.target, args);this.close();
        }return returnData;
    }
}interface IMessage{  //傳統(tǒng)代理設(shè)計必須有接口void send();  //業(yè)務(wù)方法}class MessageReal implements IMessage {@Overridepublic void send() {
        System.out.println("【發(fā)送消息】www.mldn.cn");
    }
}

執(zhí)行結(jié)果:
什么是靜態(tài)代理設(shè)計模式

如果認(rèn)真觀察系統(tǒng)中提供的Proxy.newProxyInstance()方法,會發(fā)現(xiàn)該方法會使用大量的底層機制來進行代理對象的動態(tài)創(chuàng)建,所有的代理類是符合所有相關(guān)功能需求的操作功能類,它不再代表具體的接口,這樣在處理時就必須依賴于類加載器與接口進行代理對象的偽造。

到此,相信大家對“什么是靜態(tài)代理設(shè)計模式”有了更深的了解,不妨來實際操作一番吧!這里是億速云網(wǎng)站,更多相關(guān)內(nèi)容可以進入相關(guān)頻道進行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI