摘要:使用面向切面編程來快速的創(chuàng)建職責鏈的具體概念可以參考裝飾者模式實現(xiàn)職責鏈簡單又巧妙,但這種把函數(shù)疊在一起的方式,同時也疊加了函數(shù)的作用域,如果鏈條太長的話,也會對性能造成太大的影響。在開發(fā)中,職責鏈模式是最容易被忽視的模式之一。
聲明:這個系列為閱讀《JavaScript設計模式與開發(fā)實踐》 ----曾探@著一書的讀書筆記
1.職責鏈模式的定義
2.
2.1 簡單職責鏈模式
2.2職責鏈重構(gòu)上面的代碼
2.3靈活的拆分職責鏈節(jié)點
3.異步職責鏈
4.職責鏈模式的優(yōu)缺點:
5.使用AOP(面向切面編程)來快速的創(chuàng)建職責鏈
總結(jié):
1.職責鏈模式的定義2. 2.1 簡單職責鏈模式使多個對象都有機會處理請求,從而避免請求的發(fā)送者和接收者之間的耦合關系,將這些對象連成一條鏈,并沿著這條鏈傳遞該請求,直到有一個對象處理它為止。
故事背景:用戶可以支付定金購買手機,并且可以獲得優(yōu)惠券。沒有支付定金的就是普通用戶,進入普通購買模式,沒有優(yōu)惠券,且?guī)齑娌蛔愕那闆r下不一定能夠買到手機。
/** * * @param orderType 訂單類型 * @param pay 用戶是否已經(jīng)支付過定金 true or false * @param stock 表示手機的庫存量 */ var order = function (orderType, pay, stock){ if (orderType === 1) { if (pay === true) { console.log("500定金預購,得到100元優(yōu)惠券"); } else { if (stock > 0) { console.log("普通購買,沒有優(yōu)惠券"); } else { console.log("手機庫存不足"); } } } else if (orderType === 2) { if (pay === true) { console.log("200定金預購,得到50元優(yōu)惠券"); } else { if (stock > 0) { console.log("普通購買,沒有優(yōu)惠券"); } else { console.log("手機庫存不足"); } } } else if (orderType === 3) { if (stock > 0) { console.log("普通購買,沒有優(yōu)惠券"); } else { console.log("手機庫存不足"); } } };2.2職責鏈重構(gòu)上面的代碼
主要通過拆分功能語句,來使用職責鏈重構(gòu):
//500元訂單 var order500 = function (orderType, pay, stock){ if (orderType === 1 && pay === true) { console.log("500定金預購,得到100元優(yōu)惠券"); } else { order200(orderType, pay, stock); //將請求傳遞給200 } }; //200元訂單 var order200 = function (orderType, pay, stock){ if (orderType === 2 && pay === true) { console.log("200定金預購,得到50元優(yōu)惠券"); } else { order(orderType, pay, stock); } }; //普通購買訂單 var order = function (orderType, pay, stock){ if (stock>0) { console.log("普通購買,沒有優(yōu)惠券"); } else { console.log("手機庫存不足"); } }; //測試調(diào)用 order500(1,true,500); order500(3,false,0);
總結(jié):
上面的代碼違反了開放-封閉的原則,請求在鏈條中傳遞的順序非常僵硬,傳遞請求的代碼被耦合在了業(yè)務函數(shù)中:
var order500 = function (orderType, pay, stock){ if (orderType === 1 && pay === true) { console.log("500定金預購,得到100元優(yōu)惠券"); } else { order200(orderType, pay, stock); //將請求傳遞給200 } };2.3靈活的拆分職責鏈節(jié)點
為什么要拆分職責鏈的節(jié)點,因為某天需要添加新的職責,就需要修改業(yè)務代碼(要修改的話,就需要先去了解他,熟悉它,花費大量的時間)。這顯然不是每一個人所需要的。
//500元訂單 var order500 = function (orderType, pay, stock){ if (orderType === 1 && pay === true) { console.log("500定金預購,得到100元優(yōu)惠券"); } else { return "nextSuccessor"; } }; //200元訂單 var order200 = function (orderType, pay, stock){ if (orderType === 2 && pay === true) { console.log("200定金預購,得到50元優(yōu)惠券"); } else { return "nextSuccessor"; } }; //普通購買訂單 var order = function (orderType, pay, stock){ if (stock>0) { console.log("普通購買,沒有優(yōu)惠券"); } else { console.log("手機庫存不足"); } }; var Chain=function (fn){ this.fn=fn; this.successor=null; }; Chain.prototype.setNextSuccessor=function (successor){ return this.successor=successor; }; Chain.prototype.passRequest=function(){ var ret= this.fn.apply(this,arguments); if(ret==="nextSuccessor"){ return this.successor && this.successor.passRequest.apply(this.successor,arguments); } return ret; }; var chainOrder500=new Chain(order500()); var chainOrder200=new Chain(order200()); var chainOrderNormal=new Chain(order()); chainOrder500.setNextSuccessor(chainOrder200); chainOrder200.setNextSuccessor(chainOrderNormal); chainOrder500.passRequest(1,true,500); chainOrder500.passRequest(2,true,500); chainOrder500.passRequest(1,false,0);
加入某天網(wǎng)站添加了300元定金購買的職責,我只需要添加特定的節(jié)點就可以了:
//300元訂單 var order300=function (){ }; var chainOrder300=new Chain(order300()); chainOrder500.setNextSuccessor(chainOrder300); chainOrder500.setNextSuccessor(chainOrder200);
這樣的話只需要編寫簡單的功能函數(shù),改變職責鏈中的相關節(jié)點的順序即可。
3.異步職責鏈上面的職責鏈代碼中,每個節(jié)點函數(shù)同步返回一個特定的值nextSuccessor,來表示是否把請求傳遞給下一個節(jié)點。而現(xiàn)實開發(fā)中會遇到一些異步的問題,比如在一個節(jié)點中發(fā)起一個ajax異步請求,異步請求的結(jié)果才能決定是否繼續(xù)在職責鏈中passRequest。
可以給Chain類添加一個原型方法Chain.prototype.next,表示手動傳遞請求給職責鏈中的下一個節(jié)點:
Chain.prototype.next=function(){ return this.successor && this.successor.passRequest.apply(this.successor,arguments); }; //異步職責鏈的例子 var fn1=new Chain(function (){ console.log(1); return "nextSuccessor"; }); var fn2=new Chain(function (){ console.log(2); var self=this; setTimeout(function (){ self.next(); },1000); }); var fn3=new Chain(function (){ console.log(3); }); fn1.setNextSuccessor(fn2).setNextSuccessor(fn3); fn1.passRequest();4.職責鏈模式的優(yōu)缺點:
優(yōu)點:
職責鏈最大的優(yōu)點就是解耦了請求發(fā)送者和N個接收者之間的復雜關系。
職責鏈可以手動指定起始節(jié)點,請求并不是非得從鏈中的第一個節(jié)點開始傳遞。
缺點:
不能保證某個請求一定會被鏈中的節(jié)點處理,這種情況可以在鏈尾增加一個保底的接受者節(jié)點來處理這種即將離開鏈尾的請求。
使程序中多了很多節(jié)點對象,可能再一次請求的過程中,大部分的節(jié)點并沒有起到實質(zhì)性的作用。他們的作用僅僅是讓請求傳遞下去,從性能當面考慮,要避免過長的職責鏈到來的性能損耗。
5.使用AOP(面向切面編程)來快速的創(chuàng)建職責鏈AOP的具體概念可以參考裝飾者模式
Function.prototype.after=function(fn){ var self=this; return function(){ var ret=self.apply(this,arguments); if(ret==="nextSuccessor"){ return fn.apply(this,arguments); } return ret; } }; var order=order500yuan.after(order200yuan).after(orderNormal); order(1,true,500); order(1,false,500);
AOP實現(xiàn)職責鏈簡單又巧妙,但這種把函數(shù)疊在一起的方式,同時也疊加了函數(shù)的作用域,如果鏈條太長的話,也會對性能造成太大的影響。
總結(jié):職責鏈模式最大的優(yōu)點:請求發(fā)送者只需要知道鏈中的第一個節(jié)點,從而弱化了發(fā)送者和一組接收者之前的強聯(lián)系。
在JavaScript開發(fā)中,職責鏈模式是最容易被忽視的模式之一。但是只要運用得當,職責鏈模式可以很好的幫助我們管理代碼,降低發(fā)起請求的對象和處理請求的對象之間的耦合性。且職責鏈中節(jié)點的數(shù)量和數(shù)序是可以自由變化的??梢栽谶\行時決定鏈中包含哪些節(jié)點。
無論是作用域鏈,原型鏈,還是DOM節(jié)點中的事件冒泡,我們都能從中找到職責鏈模式的影子。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/90839.html
摘要:代理模式,迭代器模式,單例模式,裝飾者模式最少知識原則一個軟件實體應當盡可能少地與其他實體發(fā)生相互作用。迭代器模式可以將迭代的過程從業(yè)務邏輯中分離出來,在使用迭代器模式之后,即不用關心對象內(nèi)部構(gòu)造也可以按順序訪問其中的每個元素。 接手項目越來越復雜的時候,有時寫完一段代碼,總感覺代碼還有優(yōu)化的空間,卻不知道從何處去下手。設計模式主要目的是提升代碼可擴展性以及可閱讀性。 本文主要以例子的...
摘要:裝飾者模式定義裝飾者模式能夠在不改變對象自身的基礎上,在程序運行期間給對像動態(tài)的添加職責。與繼承相比,裝飾者是一種更輕便靈活的做法。 裝飾者模式 定義 : 裝飾者(decorator)模式能夠在不改變對象自身的基礎上,在程序運行期間給對像動態(tài)的添加職責。與繼承相比,裝飾者是一種更輕便靈活的做法。 在不改變對象自身的基礎上,在程序運行期間給對象動態(tài)地添加一些額外職責 特點 : 可以動態(tài)的...
摘要:訂閱模式的一個典型的應用就是后面會寫一篇相關的讀書筆記。享元模式享元模式的核心思想是對象復用,減少對象數(shù)量,減少內(nèi)存開銷。適配器模式對目標函數(shù)進行數(shù)據(jù)參數(shù)轉(zhuǎn)化,使其符合目標函數(shù)所需要的格式。 設計模式 單例模式 JS的單例模式有別于傳統(tǒng)面向?qū)ο笳Z言的單例模式,js作為一門無類的語言。使用全局變量的模式來實現(xiàn)單例模式思想。js里面的單例又分為普通單例和惰性單例,惰性單例指的是只有這個實例...
摘要:前言設計模式幾十種,閱讀了設計模式與開發(fā)實踐這本書后,個人感覺就是圍繞對象來設計的,發(fā)現(xiàn)日常寫代碼能用上的并不多,下面是常用的幾種設計模式。前端服務端可以參考我的另一個倉庫地址,一個簡單的實時聊天參考來自設計模式與開發(fā)實踐讀書筆記 前言 設計模式幾十種,閱讀了《JavaScript設計模式與開發(fā)實踐》這本書后,個人感覺js就是圍繞對象來設計的,發(fā)現(xiàn)日常寫代碼能用上的并不多,下面是常用的...
摘要:想一想,這個和我們的迭代器模式有著異曲同工的妙處,迭代器模式同樣也是遍歷選出最優(yōu)解,但是相比而言,職責鏈模式的直觀性個書寫的幸福感是遠遠超過迭代器模式的。 職責鏈模式其實很好理解,由于一個鏈字出賣了它的靈魂。我們可以從這個字得到很大的提示。首先這個模式一定有傳遞性,而且,節(jié)點是可以重復拼接的,并且每個節(jié)點都具有一定的過濾功能,一定的職責。 是不是想起了組合模式里的一些內(nèi)容呢? 是的,他...
閱讀 2554·2023-04-26 00:57
閱讀 925·2021-11-25 09:43
閱讀 2229·2021-11-11 16:55
閱讀 2241·2019-08-30 15:53
閱讀 3604·2019-08-30 15:52
閱讀 1472·2019-08-30 14:10
閱讀 3389·2019-08-30 13:22
閱讀 1221·2019-08-29 11:18