Java設計模式之責任鏈模式
設計者往往會用攔截器去代替動態(tài)代理,然后將攔截器的接口提供給開發(fā)者,從而簡化開發(fā)者的開發(fā)難度,但是攔截器可能有多個。舉個例子,一個程序員需要請假一周,如果把請假申請單看成一個對象,那么它需要經(jīng)過項目經(jīng)理、部門經(jīng)理、人事等多個角色的審批,每個角色都有機會通過攔截這個申請單進行審批或者修改。這事就要考慮提供項目經(jīng)理、部門經(jīng)理和人事的處理邏輯,所以需要提供3個攔截器,二傳遞的則是請假申請單。
當一個對象在一條鏈上被多個攔截器處理(攔截器也可以選擇不攔截處理它)時,我們把這樣的設計模式成為責任鏈模式,它用于一個對象在多個角色中傳遞的場景。還是剛才的例子,申請單走到項目經(jīng)理,經(jīng)理可能把申請時間“一周”改為“5天”,從而影響了后面的審批,后面的審批都要根據(jù)前面的結果進行。這個時候可以考慮用層層代理來實現(xiàn),就是當申請單(target)走到項目經(jīng)理處,使用第一個動態(tài)代理proxy1,。當它走到部門經(jīng)理處,部門經(jīng)理會得到一個在項目經(jīng)理的代理proxy1基礎上生成的proxy2來處理部門經(jīng)理的邏輯。當它走到人事處,會在proxy2的基礎上生成proxy3.如果還有其他角色,依次類推即可,可用下圖來描述攔截邏輯:

我們定義下面的攔截器接口:
/**
* @Auther: haozz
* @Date: 2018/5/27 22:15
* @Description:攔截器接口
**/
public interface Interceptor {
boolean before(Object proxy, Object target, Method method,Object[] args);
void around(Object proxy,Object target,Method method,Object[] args);
void after(Object proxy,Object target,Method method,Object[] args);
}
再定義3個攔截器:
/**
* @Auther: haozz
* @Date: 2018/5/27 22:19
* @Description:攔截器1
**/
public class Interceptor1 implements Interceptor{
@Override
public boolean before(Object proxy, Object target, Method method, Object[] args) {
System.out.print("[攔截器1]的before方法");
return true;
}
@Override
public void around(Object proxy, Object target, Method method, Object[] args) {
}
@Override
public void after(Object proxy, Object target, Method method, Object[] args) {
System.out.print("[攔截器1]的after方法");
}
}
/**
* @Auther: haozz
* @Date: 2018/5/27 22:19
* @Description:攔截器2
**/
public class Interceptor2 implements Interceptor{
@Override
public boolean before(Object proxy, Object target, Method method, Object[] args) {
System.out.print("[攔截器2]的before方法");
return true;
}
@Override
public void around(Object proxy, Object target, Method method, Object[] args) {
}
@Override
public void after(Object proxy, Object target, Method method, Object[] args) {
System.out.print("[攔截器2]的after方法");
}
}
/**
* @Auther: haozz
* @Date: 2018/5/27 22:19
* @Description:攔截器3
**/
public class Interceptor3 implements Interceptor{
@Override
public boolean before(Object proxy, Object target, Method method, Object[] args) {
System.out.print("[攔截器3]的before方法");
return true;
}
@Override
public void around(Object proxy, Object target, Method method, Object[] args) {
}
@Override
public void after(Object proxy, Object target, Method method, Object[] args) {
System.out.print("[攔截器3]的after方法");
}
}
我們使用上一篇(Java動態(tài)代理之攔截器的應用)中用到的InterceptorJdkProxy類,測試一下這段代碼。如下:
@Test
public void MyTest(){
HelloWorld proxy1 = (HelloWorld) InterceptorJdkProxy.bind(new HelloWorldImpl(),"com.csdn.blog.impl.Interceptor1");
HelloWorld proxy2 = (HelloWorld) InterceptorJdkProxy.bind(proxy1,"com.csdn.blog.impl.Interceptor2");
HelloWorld proxy3 = (HelloWorld) InterceptorJdkProxy.bind(proxy2,"com.csdn.blog.impl.Interceptor3");
proxy3.sayHelloWorld();
}
運行這段diamante后得到這樣的結果,請注意觀察其方法的執(zhí)行順序:
[攔截器3]的before方法
[攔截器2]的before方法
[攔截器1]的before方法
Hello World
[攔截器1]的after方法
[攔截器2]的after方法
[攔截器3]的after方法
before方法按照最后一個攔截器到第一個攔截器的加載順序運行,而after方法則按照從第一個攔截器到最后一個攔截器的加載順序運行。
從代碼中可見,責任鏈模式的優(yōu)點在于我們可以在傳遞鏈上加入新的攔截器,增加攔截邏輯,其缺點是會增加代理和反射,而代理和反射的性能不高。
總結
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。如果你想了解更多相關內(nèi)容請查看下面相關鏈接
相關文章
基于Java語言MD5加密Base64轉(zhuǎn)換方法
這篇文章主要為大家詳細介紹了基于Java語言的MD5加密Base64轉(zhuǎn)換方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-09-09
Java元素排序Comparable與Comparator的區(qū)別
這篇文章主要介紹了Java元素排序Comparable與Comparator的區(qū)別,二者都是頂級的接口,但擁有的方法和用法是不同的,下面我們分別來看看具體是怎樣的區(qū)別吧2022-05-05
SpringBoot+mybatis+Vue實現(xiàn)前后端分離項目的示例
本文主要介紹了SpringBoot+mybatis+Vue實現(xiàn)前后端分離項目的示例,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-12-12
Apache Commons Math3探索之多項式曲線擬合實現(xiàn)代碼
這篇文章主要介紹了Apache Commons Math3探索之多項式曲線擬合實現(xiàn)代碼,小編覺得挺不錯的,這里分享給大家,供需要的朋友參考。2017-10-10

