責(zé)任鏈模式是一種對(duì)象的行為模式铃芦。在責(zé)任鏈模式里,很多對(duì)象由每一個(gè)對(duì)象對(duì)其下家的引用而連接起來(lái)形成一條鏈。請(qǐng)求在這個(gè)鏈上傳遞咳蔚,直到鏈上的某一個(gè)對(duì)象決定處理此請(qǐng)求。發(fā)出這個(gè)請(qǐng)求的客戶端并不知道鏈上的哪一個(gè)對(duì)象最終處理這個(gè)請(qǐng)求搔驼,這使得系統(tǒng)可以在不影響客戶端的情況下動(dòng)態(tài)地重新組織和分配責(zé)任
我對(duì)于責(zé)任鏈的理解就是如果if else if else if ... else 這樣的邏輯代碼寫(xiě)多了難以維護(hù)谈火,而且當(dāng)我們?cè)僭黾右粋€(gè)邏輯條件的時(shí)候不能很方便的擴(kuò)展。
使用場(chǎng)景
來(lái)考慮這樣一個(gè)功能:申請(qǐng)聚餐費(fèi)用的管理舌涨。
很多公司都是這樣的福利糯耍,就是項(xiàng)目組或者是部門(mén)可以向公司申請(qǐng)一些聚餐費(fèi)用,用于組織項(xiàng)目組成員或者是部門(mén)成員進(jìn)行聚餐活動(dòng)囊嘉。
申請(qǐng)聚餐費(fèi)用的大致流程一般是:由申請(qǐng)人先填寫(xiě)申請(qǐng)單温技,然后交給領(lǐng)導(dǎo)審批,如果申請(qǐng)批準(zhǔn)下來(lái)哗伯,領(lǐng)導(dǎo)會(huì)通知申請(qǐng)人審批通過(guò)荒揣,然后申請(qǐng)人去財(cái)務(wù)領(lǐng)取費(fèi)用,如果沒(méi)有批準(zhǔn)下來(lái)焊刹,領(lǐng)導(dǎo)會(huì)通知申請(qǐng)人審批未通過(guò)系任,此事也就此作罷。
不同級(jí)別的領(lǐng)導(dǎo)虐块,對(duì)于審批的額度是不一樣的俩滥,比如,項(xiàng)目經(jīng)理只能審批500元以內(nèi)的申請(qǐng)贺奠;部門(mén)經(jīng)理能審批1000元以內(nèi)的申請(qǐng)霜旧;而總經(jīng)理可以審核任意額度的申請(qǐng)。
也就是說(shuō)儡率,當(dāng)某人提出聚餐費(fèi)用申請(qǐng)的請(qǐng)求后挂据,該請(qǐng)求會(huì)經(jīng)由項(xiàng)目經(jīng)理、部門(mén)經(jīng)理儿普、總經(jīng)理之中的某一位領(lǐng)導(dǎo)來(lái)進(jìn)行相應(yīng)的處理崎逃,但是提出申請(qǐng)的人并不知道最終會(huì)由誰(shuí)來(lái)處理他的請(qǐng)求,一般申請(qǐng)人是把自己的申請(qǐng)?zhí)峤唤o項(xiàng)目經(jīng)理眉孩,或許最后是由總經(jīng)理來(lái)處理他的請(qǐng)求个绍。
可以使用責(zé)任鏈模式來(lái)實(shí)現(xiàn)上述功能:當(dāng)某人提出聚餐費(fèi)用申請(qǐng)的請(qǐng)求后勒葱,該請(qǐng)求會(huì)在 目經(jīng)理—〉部門(mén)經(jīng)理—〉總經(jīng)理 這樣一條領(lǐng)導(dǎo)處理鏈上進(jìn)行傳遞,發(fā)出請(qǐng)求的人并不知道誰(shuí)會(huì)來(lái)處理他的請(qǐng)求巴柿,每個(gè)領(lǐng)導(dǎo)會(huì)根據(jù)自己的職責(zé)范圍凛虽,來(lái)判斷是處理請(qǐng)求還是把請(qǐng)求交給更高級(jí)別的領(lǐng)導(dǎo),只要有領(lǐng)導(dǎo)處理了广恢,傳遞就結(jié)束了凯旋。
需要把每位領(lǐng)導(dǎo)的處理獨(dú)立出來(lái),實(shí)現(xiàn)成單獨(dú)的職責(zé)處理對(duì)象袁波,然后為它們提供一個(gè)公共的瓦阐、抽象的父職責(zé)對(duì)象,這樣就可以在客戶端來(lái)動(dòng)態(tài)地組合職責(zé)鏈篷牌,實(shí)現(xiàn)不同的功能要求了。
源代碼
抽象處理者角色類
public abstract class Handler {
/**
* 持有下一個(gè)處理請(qǐng)求的對(duì)象
*/
protected Handler successor = null;
/**
* 取值方法
*/
public Handler getSuccessor() {
return successor;
}
/**
* 設(shè)置下一個(gè)處理請(qǐng)求的對(duì)象
*/
public void setSuccessor(Handler successor) {
this.successor = successor;
}
/**
* 處理聚餐費(fèi)用的申請(qǐng)
* @param user 申請(qǐng)人
* @param fee 申請(qǐng)的錢(qián)數(shù)
* @return 成功或失敗的具體通知
*/
public abstract String handleFeeRequest(String user , double fee);
}
具體處理者角色
public class ProjectManager extends Handler {
@Override
public String handleFeeRequest(String user, double fee) {
String str = "";
//項(xiàng)目經(jīng)理權(quán)限比較小踏幻,只能在500以內(nèi)
if(fee < 500)
{
//為了測(cè)試枷颊,簡(jiǎn)單點(diǎn),只同意張三的請(qǐng)求
if("張三".equals(user))
{
str = "成功:項(xiàng)目經(jīng)理同意【" + user + "】的聚餐費(fèi)用该面,金額為" + fee + "元";
}else
{
//其他人一律不同意
str = "失斬裁纭:項(xiàng)目經(jīng)理不同意【" + user + "】的聚餐費(fèi)用,金額為" + fee + "元";
}
}else
{
//超過(guò)500隔缀,繼續(xù)傳遞給級(jí)別更高的人處理
if(getSuccessor() != null)
{
return getSuccessor().handleFeeRequest(user, fee);
}
}
return str;
}
}
public class DeptManager extends Handler {
@Override
public String handleFeeRequest(String user, double fee) {
String str = "";
//部門(mén)經(jīng)理的權(quán)限只能在1000以內(nèi)
if(fee < 1000)
{
//為了測(cè)試题造,簡(jiǎn)單點(diǎn),只同意張三的請(qǐng)求
if("張三".equals(user))
{
str = "成功:部門(mén)經(jīng)理同意【" + user + "】的聚餐費(fèi)用猾瘸,金額為" + fee + "元";
}else
{
//其他人一律不同意
str = "失斀缗狻:部門(mén)經(jīng)理不同意【" + user + "】的聚餐費(fèi)用,金額為" + fee + "元";
}
}else
{
//超過(guò)1000牵触,繼續(xù)傳遞給級(jí)別更高的人處理
if(getSuccessor() != null)
{
return getSuccessor().handleFeeRequest(user, fee);
}
}
return str;
}
}
public class GeneralManager extends Handler {
@Override
public String handleFeeRequest(String user, double fee) {
String str = "";
//總經(jīng)理的權(quán)限很大淮悼,只要請(qǐng)求到了這里,他都可以處理
if(fee >= 1000)
{
//為了測(cè)試揽思,簡(jiǎn)單點(diǎn)袜腥,只同意張三的請(qǐng)求
if("張三".equals(user))
{
str = "成功:總經(jīng)理同意【" + user + "】的聚餐費(fèi)用,金額為" + fee + "元";
}else
{
//其他人一律不同意
str = "失敹ず埂:總經(jīng)理不同意【" + user + "】的聚餐費(fèi)用羹令,金額為" + fee + "元";
}
}else
{
//如果還有后繼的處理對(duì)象,繼續(xù)傳遞
if(getSuccessor() != null)
{
return getSuccessor().handleFeeRequest(user, fee);
}
}
return str;
}
}
客戶端類
public class Client {
public static void main(String[] args) {
//先要組裝責(zé)任鏈
Handler h1 = new GeneralManager();
Handler h2 = new DeptManager();
Handler h3 = new ProjectManager();
h3.setSuccessor(h2);
h2.setSuccessor(h1);
//開(kāi)始測(cè)試
String test1 = h3.handleFeeRequest("張三", 300);
System.out.println("test1 = " + test1);
String test2 = h3.handleFeeRequest("李四", 300);
System.out.println("test2 = " + test2);
System.out.println("---------------------------------------");
String test3 = h3.handleFeeRequest("張三", 700);
System.out.println("test3 = " + test3);
String test4 = h3.handleFeeRequest("李四", 700);
System.out.println("test4 = " + test4);
System.out.println("---------------------------------------");
String test5 = h3.handleFeeRequest("張三", 1500);
System.out.println("test5 = " + test5);
String test6 = h3.handleFeeRequest("李四", 1500);
System.out.println("test6 = " + test6);
}
}
運(yùn)行結(jié)果
test1 = 成功:項(xiàng)目經(jīng)理同意【張三】的聚餐費(fèi)用损痰,金額為300.0元
test2 = 失敻3蕖:項(xiàng)目經(jīng)理不同意【李四】的聚餐費(fèi)用,金額為300.0元
---------------------------------------
test3 = 成功:部門(mén)經(jīng)理同意【張三】的聚餐費(fèi)用徐钠,金額為700.0元
test4 = 失敯┕簟:部門(mén)經(jīng)理不同意【李四】的聚餐費(fèi)用,金額為700.0元
---------------------------------------
test5 = 成功:總經(jīng)理同意【張三】的聚餐費(fèi)用,金額為1500.0元
test6 = 失斚园荨:總經(jīng)理不同意【李四】的聚餐費(fèi)用衡奥,金額為1500.0元