Java的代理模式你真的了解嗎
代理模式原理解析
代理模式(Proxy Design Pattern),它在不改變原始類(或者叫被代理類)代碼的情況下,通過引入代理類來給原始類附加功能。
public class UserController {
//...省略其他屬性和方法...
private MetricsCollector metricsCollector; // 依賴注入
public UserVo login(String telephone, String password) {
long startTimestamp = System.currentTimeMillis();
// ... 省略login邏輯...
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
RequestInfo requestInfo = new RequestInfo("login", responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
//...返回UserVo數據...
}
public UserVo register(String telephone, String password) {
long startTimestamp = System.currentTimeMillis();
// ... 省略register邏輯...
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
RequestInfo requestInfo = new RequestInfo("register", responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
//...返回UserVo數據...
}
}
上面代碼有兩個問題,第一,性能計數器框架代碼侵入到業(yè)務代碼中,跟業(yè)務高度耦合。如果需要替換這個框架,成本比較大。第二,收集接口請求的代碼跟業(yè)務代碼無關,不應該放到一個類中。業(yè)務類最好職責更加單一,只負責業(yè)務處理。
現在改成代理模式:
public interface IUserController {
UserVo login(String telephone, String password);
UserVo register(String telephone, String password);
}
public class UserController implements IUserController {
//...省略其他屬性和方法...
@Override
public UserVo login(String telephone, String password) {
//...省略login邏輯...
//...返回UserVo數據...
}
@Override
public UserVo register(String telephone, String password) {
//...省略register邏輯...
//...返回UserVo數據...
}
}
public class UserControllerProxy implements IUserController {
private MetricsCollector metricsCollector;
private UserController userController;
public UserControllerProxy(UserController userController) {
this.userController = userController;
this.metricsCollector = new MetricsCollector();
}
@Override
public UserVo login(String telephone, String password) {
long startTimestamp = System.currentTimeMillis();
// 委托
UserVo userVo = userController.login(telephone, password);
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
RequestInfo requestInfo = new RequestInfo("login", responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
return userVo;
}
@Override
public UserVo register(String telephone, String password) {
long startTimestamp = System.currentTimeMillis();
UserVo userVo = userController.register(telephone, password);
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
RequestInfo requestInfo = new RequestInfo("register", responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
return userVo;
}
}
//UserControllerProxy使用舉例
//因為原始類和代理類實現相同的接口,是基于接口而非實現編程
//將UserController類對象替換為UserControllerProxy類對象,不需要改動太多代碼
IUserController userController = new UserControllerProxy(new UserController());
按照基于接口而非實現編程的設計思想,將原始類對象替換為代理類對象的時候,剛剛的代理模式代碼實現中,代理類和原始類需要實現相同的接口。但是如果原始類沒有定義接口,并且原始類代碼并不是我們開發(fā)維護的,也不能重新定義一個接口,這種情況下,如何實現代理模式呢?
對于這種外部類的擴展,一般都采用繼承的方式。讓代理類繼承原始類,然后擴展附加功能。代碼如下:
public class UserControllerProxy extends UserController {
private MetricsCollector metricsCollector;
public UserControllerProxy() {
this.metricsCollector = new MetricsCollector();
}
public UserVo login(String telephone, String password) {
long startTimestamp = System.currentTimeMillis();
UserVo userVo = super.login(telephone, password);
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
RequestInfo requestInfo = new RequestInfo("login", responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
return userVo;
}
public UserVo register(String telephone, String password) {
long startTimestamp = System.currentTimeMillis();
UserVo userVo = super.register(telephone, password);
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
RequestInfo requestInfo = new RequestInfo("register", responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
return userVo;
}
}
//UserControllerProxy使用舉例
UserController userController = new UserControllerProxy();
動態(tài)代理的原理解析
另外,剛剛的代碼實現還是有點問題。一方面,我們需要在代理類中,將原始類中的所有方法,都要重新實現一遍,而且為每個方法都附加相似的代碼邏輯。另一方面,如果要添加功能的類不止有一個,我們需要針對每個類都創(chuàng)建一個代理類。
如果有50個要添加附加功能的原始類,那我們就要創(chuàng)建50個對應的代理類,導致項目中的類個數成倍增加,提高維護成本。而且每個代理類的代碼都有點重復,增加開發(fā)成本,這種問題該如何解決?
可以使用動態(tài)代理(Dynamic Proxy),Java動態(tài)代理實現。
public class MetricsCollectorProxy {
private MetricsCollector metricsCollector;
public MetricsCollectorProxy() {
this.metricsCollector = new MetricsCollector();
}
public Object createProxy(Object proxiedObject) {
Class<?>[] interfaces = proxiedObject.getClass().getInterfaces();
DynamicProxyHandler handler = new DynamicProxyHandler(proxiedObject);
return Proxy.newProxyInstance(proxiedObject.getClass().getClassLoader(), interfaces, handler);
}
private class DynamicProxyHandler implements InvocationHandler {
private Object proxiedObject;
public DynamicProxyHandler(Object proxiedObject) {
this.proxiedObject = proxiedObject;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
long startTimestamp = System.currentTimeMillis();
Object result = method.invoke(proxiedObject, args);
long endTimeStamp = System.currentTimeMillis();
long responseTime = endTimeStamp - startTimestamp;
String apiName = proxiedObject.getClass().getName() + ":" + method.getName();
RequestInfo requestInfo = new RequestInfo(apiName, responseTime, startTimestamp);
metricsCollector.recordRequest(requestInfo);
return result;
}
}
}
//MetricsCollectorProxy使用舉例
MetricsCollectorProxy proxy = new MetricsCollectorProxy();
IUserController userController = (IUserController) proxy.createProxy(new UserController());
代理模式的應用場景
代理模式的應用場景非常多,下面舉例說一些比較常見的用法。
- 業(yè)務系統(tǒng)的非功能性需求開發(fā)
比如在業(yè)務系統(tǒng)中的一些非功能性需求,監(jiān)控、統(tǒng)計、鑒權、限流、事物、冪等、日志。這些附加功能與業(yè)務系統(tǒng)功能解耦,放到代理類中統(tǒng)一處理,讓程序員只需要關注業(yè)務開發(fā)。
- 代理模式在PRC、緩存中的應用
實際上,RPC框架也可以看作一種代理模式,GoF的設計模式一書中把它稱作遠程代理,通過遠程代理,將網絡通信、數據編碼解碼等細節(jié)隱藏起來。客戶端在使用RPC服務的時候,就像使用本地函數一樣,無需了解跟服務器交互的細節(jié)。
代理在緩存中的應用,假設要開發(fā)一個接口請求的緩存功能,對于某些接口請求,輸入參數相同,在設定的過期時間內,直接返回緩存結果,不用重新進行邏輯業(yè)務處理
到此這篇關于Java的代理模式你真的了解嗎的文章就介紹到這了,更多相關Java代理模式內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
一篇文章帶你了解jdk1.8新特性--為什么使用lambda表達式
Lambda是一個匿名函數,我們可以把Lambda表達式理解為是一段可以傳遞的代碼,本篇文章就帶你了解,希望能給你帶來幫助2021-08-08
Spring中獲取Bean對象的三種注入方式與兩種注入方法詳解
平常的Java開發(fā)中程序員在某個類中需要依賴其它類的方法,下面這篇文章主要給大家介紹了關于Spring中獲取Bean對象的三種注入方式與兩種注入方法的相關資料,文中通過圖文介紹的非常詳細,需要的朋友可以參考下2023-03-03
IntelliJ IDEA Java項目手動添加依賴 jar 包的方法(圖解)
這篇文章主要介紹了IntelliJ IDEA Java項目手動添加依賴 jar 包,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-04-04

