Java設計模式之責任鏈模式的示例詳解
責任鏈模式是將鏈中的每一個節(jié)點看做是一個對象,每個節(jié)點處理的請求均不相同,且內(nèi)部自動維護下一個節(jié)點對象,當一個請求從鏈式的首段發(fā)出時,會沿著鏈的路徑依次傳遞給每一個節(jié)點對象,直至有對象處理這個請求位置,屬于行為模式。
這里需要注意的是每個節(jié)點都能對對象進行一定的處理(也可以不處理),處理完成之后節(jié)點再進行判斷還要進行后續(xù)處理還是說傳遞給下一個節(jié)點。
應用場景
首先舉一個日常的例子,比如我們申請開發(fā)票,首先我們要寫好報銷單,首先要你的部門領導審批,部門領導審批不通過直接打回,審批通過再由公司的總經(jīng)理審批這里審批通過才算成審批完成。這種情況就很適合使用責任鏈模式。
總結(jié)一下責任鏈主要適用一下幾種情況:
- 多個對象可以處理同一個請求,但是具體由那個對象處理完成則在運行時決定。
- 不明確指定接收者的情況下,向多個對象中的一個提交一個請求
可以看一下責任鏈模式的通用UML類圖:
通過類圖可以看到總共包含以下角色:
- 抽象處理者:主要是定義處理請求的方法以及維護下一個處理結(jié)點的對象的引用
- 具體處理者:處理的具體實現(xiàn)
責任鏈的精髓在于將很多處理節(jié)點行成個鏈式結(jié)構(gòu),并允許結(jié)點自身決定是否進行處理或者轉(zhuǎn)發(fā)。
實際代碼案例
下面舉一個我們在開發(fā)時經(jīng)常會遇到的一種情況:登錄 比如我們開發(fā)一個管理系統(tǒng)在登錄的時候往往我們會先判斷客戶端傳遞的賬號及密碼是否為空但凡有一個是空肯定是不能繼續(xù)往下走的,然后就是根據(jù)用戶賬號密碼拿到用戶的所有信息,如果能拿到繼續(xù)周下一步,拿不到則是報錯提示用戶不存在,到下一步又會判斷當前用戶的權(quán)限。
無模式情況下的代碼
private String login(String username,String password) { if(username == null || password == null) { return "賬戶或者密碼為null"; } User user = queryUserInfo(username, password); if(user == null) { return "找不到用戶"; } if(!Objects.equals(user.getRoleName(), "超管")){ return "沒有權(quán)限"; } return "登錄成功"; } private User queryUserInfo(String username,String password){ if(Objects.equals(username, "土豆") && Objects.equals(password, "666666")) { return new User(username,password,"超管"); }else if(Objects.equals(username, "土豆2號") && Objects.equals(password, "666666")){ return new User(username,password,"普通員工"); } return null; }
發(fā)現(xiàn)判斷代碼都冗余在一個方法里面,后續(xù)改動修改都需要修改中這個方法不滿足開閉原則。
采用責任鏈模式優(yōu)化代碼
首先創(chuàng)建抽象類規(guī)定抽象方法以及維護下一個節(jié)點
public abstract class Handler { protected Handler next; public void setNext(Handler next) { this.next = next; } public abstract void doHandler(User user); }
然后就是創(chuàng)建多個實現(xiàn)邏輯的節(jié)點對象:
public class ValidatedHandler extends Handler { @Override public void doHandler(User user) { if(user.getUsername() == null || user.getPassword() == null) { System.out.println("賬戶或者密碼為null"); }else{ this.next.doHandler(user); } } }
public class UserHandler extends Handler{ @Override public void doHandler(User user) { queryUserInfo(user); if(user.getRoleName() == null){ System.out.println("沒有找到用戶"); }else{ this.next.doHandler(user); } } private static void queryUserInfo(User user){ if(Objects.equals(user.getUsername(), "土豆") && Objects.equals(user.getPassword(), "666666")) { user.setRoleName("超管"); }else if(Objects.equals(user.getUsername(), "土豆2號") && Objects.equals(user.getPassword(), "666666")){ user.setRoleName("普通員工"); } } }
public class AuthHandler extends Handler{ @Override public void doHandler(User user) { if(!Objects.equals(user.getRoleName(), "超管")){ System.out.println("沒有權(quán)限"); } System.out.println("登入成功"); } }
最后調(diào)用:
public static void main(String[] args) { User user = new User("土豆","666666"); Handler validatedHandler = new ValidatedHandler(); Handler userHandler = new UserHandler(); Handler authHandler = new AuthHandler(); validatedHandler.setNext(userHandler); userHandler.setNext(authHandler); validatedHandler.doHandler(user); }
可以看一下UML類圖:
采用建造者+責任鏈模式優(yōu)化代碼
上述的代碼發(fā)現(xiàn)維護鏈表的操作在用戶調(diào)用的那一層,鏈表的組裝過于復雜,這個時候我們可以使用建造者模式, 自動維護鏈表的組裝,調(diào)用者只需要指定鏈表的順序即可 主要修改Handler內(nèi)代碼:
public abstract class Handler { protected Handler next; public void setNext(Handler next) { this.next = next; } public abstract void doHandler(User user); public static class Builder { private Handler head; private Handler tail; public Builder addHandler(Handler handler) { if(this.head == null){ this.head = this.tail = handler; return this; } this.tail.setNext(handler); this.tail = handler; return this; } public Handler build() { return this.head; } } }
Handler.Builder builder = new Handler.Builder(); builder.addHandler(new ValidatedHandler()) .addHandler(new UserHandler()) .addHandler(new AuthHandler()); builder.build().doHandler(user);
可發(fā)現(xiàn)調(diào)用者只需要關(guān)心鏈表順序?qū)懭刖秃?/p>
責任鏈模式優(yōu)缺點
優(yōu)點:
- 請求與處理解耦
- 處理者只需要關(guān)心自己的處理邏輯即可,如果不是自己的直接轉(zhuǎn)發(fā)
- 具有鏈式傳遞功能,請求者不需要關(guān)系鏈路結(jié)構(gòu)等待結(jié)果就好
- 易于維護,可以很靈活的修改鏈路的結(jié)構(gòu)新增或者刪除,符合開閉原則
缺點:
- 會出現(xiàn)循環(huán)引用的情況
- 責任鏈太長會影響性能
到此這篇關(guān)于Java設計模式之責任鏈模式的示例詳解的文章就介紹到這了,更多相關(guān)Java責任鏈模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java.lang.NullPointerException 如何處理空指針異常的實現(xiàn)
這篇文章主要介紹了java.lang.NullPointerException 如何處理空指針異常的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-12-12Kotlin語法學習-變量定義、函數(shù)擴展、Parcelable序列化等簡單總結(jié)
這篇文章主要介紹了Kotlin語法學習-變量定義、函數(shù)擴展、Parcelable序列化等簡單總結(jié)的相關(guān)資料,需要的朋友可以參考下2017-05-05SpringSecurity退出功能實現(xiàn)的正確方式(推薦)
本文將介紹在Spring Security框架下如何實現(xiàn)用戶的"退出"logout的功能。本文通過實例代碼講解的非常詳細,具有一定的參考借鑒價值,需要的朋友參考下吧2019-11-11Maven配置項目依賴使用本地倉庫的方法匯總(小結(jié))
這篇文章主要介紹了Maven配置項目依賴使用本地倉庫的方法匯總(小結(jié)),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-07-07