代理模式(Proxy Design Pattern)在不改變原始類(或叫被代理類)代碼的情況下,通過引入代理類來給原始類附加功能。通過GPT來一探原理。 ...
引言
代理模式(Proxy Design Pattern)在不改變原始類(或叫被代理類)代碼的情況下,通過引入代理類來給原始類附加功能。
代理模式的關鍵角色包括:
- 抽象主題(Subject):定義了目標對象和代理對象的共同介面,這樣一來在任何可以使用目標對象的地方都可以使用代理對象。
- 目標對象(Real Subject):也稱為被代理對象,是具體業務邏輯的實際執行者。
- 代理對象(Proxy):負責代理目標對象,它持有對目標對象的引用,併在其自身的方法中調用目標對象的方法,同時還可以在調用前後進行一些其他的操作。
應用場景
代理模式可以應用於許多場景,以下是幾個常見的應用場景:
- 遠程代理(Remote Proxy):代理模式可以用來在客戶端和遠程對象之間建立代理對象,隱藏了實際的網路通信細節。客戶端通過代理對象調用遠程對象的方法,而無需關心網路通信的具體實現。
- 虛擬代理(Virtual Proxy):代理模式可以用來延遲載入資源密集或耗時的對象,只有當真正需要使用這些對象時,才會創建並載入真實的對象。虛擬代理可以在一定程度上提升系統性能和響應速度。
- 安全代理(Protection Proxy):代理模式可以用來控制對敏感對象的訪問許可權。代理對象可以在調用目標對象方法之前檢查許可權,如果沒有足夠的許可權,則不允許訪問。
- 緩存代理(Caching Proxy):代理模式可以用來緩存目標對象的計算結果,當相同的請求再次到達時,可以直接返回緩存的結果,避免重覆計算,提高系統性能。
- 日誌記錄代理(Logging Proxy):代理模式可以在目標對象的方法執行前後進行日誌記錄,用於跟蹤和調試系統運行過程中的操作。
- AOP(面向切麵編程):代理模式是實現AOP的基礎,可以通過代理對象在目標對象的方法執行前後插入切麵邏輯,例如日誌、事務管理等。
需要註意的是,代理模式並非適用於所有情況。在某些場景下,代理模式可能引入額外的複雜性和性能開銷,需要根據具體問題和需求來決定是否使用代理模式。
編程示例
代理模式的實現方式有多種,常見的有靜態代理和動態代理兩種形式:
- 靜態代理:在編譯時期就已經確定代理關係,代理類和目標類的關係在代碼中明確指定。
// 抽象主題
public interface Subject {
void request();
}
// 目標對象
@Slf4j
public class RealSubject implements Subject {
@Override
public void request() {
// 具體業務邏輯
LOGGER.info("開始處理請求");
}
}
@Slf4j
public class Proxy implements Subject {
private Subject realSubject;
public Proxy(Subject realSubject) {
this.realSubject = realSubject;
}
@Override
public void request() {
LOGGER.info("前置處理邏輯");
// 執行一些額外的操作
realSubject.request();
// 執行一些額外的操作
LOGGER.info("後置處理邏輯");
}
}
// 客戶端
public class Client {
public static void main(String[] args) {
Subject realSubject = new RealSubject(); // 創建目標對象
Subject proxy = new Proxy(realSubject); // 創建代理對象
proxy.request(); // 通過代理對象調用目標對象的方法
}
}
- 動態代理:在運行時動態生成代理類,無需提前編寫代理類。Java 中的動態代理主要通過
java.lang.reflect.Proxy
類和java.lang.reflect.InvocationHandler
介面實現。
// 抽象主題
public interface Subject {
void request();
}
// 目標對象
@Slf4j
public class RealSubject implements Subject {
@Override
public void request() {
// 具體業務邏輯
LOGGER.info("開始處理請求");
}
}
// InvocationHandler 實現類
public class DynamicProxy implements InvocationHandler {
private Object target;
public DynamicProxy(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 執行一些額外的操作
LOGGER.info("前置處理邏輯");
Object result = method.invoke(target, args);
// 執行一些額外的操作
LOGGER.info("後置處理邏輯");
return result;
}
}
// 客戶端
public class Client {
public static void main(String[] args) {
Subject realSubject = new RealSubject(); // 創建目標對象
InvocationHandler handler = new DynamicProxy(realSubject); // 創建 InvocationHandler 實現類
Subject proxy = (Subject) Proxy.newProxyInstance(
realSubject.getClass().getClassLoader(),
realSubject.getClass().getInterfaces(),
handler
); // 創建動態代理對象
proxy.request(); // 通過代理對象調用目標對象的方法
}
}
以上內容基於GPT創建和整理。
參考
關於作者
來自一線全棧程式員nine的八年探索與實踐,持續迭代中。歡迎關註“雨林尋北”或添加個人衛星codetrend(備註技術)。