這期內(nèi)容當(dāng)中小編將會(huì)給大家?guī)碛嘘P(guān)什么是php中責(zé)任鏈模式,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
專注于為中小企業(yè)提供做網(wǎng)站、成都網(wǎng)站建設(shè)服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)鄧州免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動(dòng)了1000多家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網(wǎng)站建設(shè)實(shí)現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。
責(zé)任鏈模式屬于行為型設(shè)計(jì)模式,將請(qǐng)求沿著處理者鏈進(jìn)行發(fā)送, 收到請(qǐng)求后, 每個(gè)處理者均可對(duì)請(qǐng)求進(jìn)行處理, 或?qū)⒄?qǐng)求傳遞給鏈上的下個(gè)處理者。鏈上的每個(gè)處理者都有自己的處理職責(zé),所以叫責(zé)任鏈模式。
場(chǎng)景
假如目前有一個(gè)系統(tǒng),現(xiàn)在我們希望對(duì)系統(tǒng)的訪問進(jìn)行限制。首先是登錄驗(yàn)證,該系統(tǒng)的所有請(qǐng)求都需要在登錄狀態(tài)下才能獲取。
過了一段時(shí)間后,老板覺得應(yīng)該加入防爬蟲程序。老板的任務(wù)最大,你立即將防爬蟲的功能加入到原本的校驗(yàn)代碼中。目前就登錄校驗(yàn)以及防爬蟲校驗(yàn),還不是很復(fù)雜。但又過了幾天,老板又覺得應(yīng)該加入限流的校驗(yàn)……。過了幾天,老板……。
關(guān)于校驗(yàn)的代碼塊已經(jīng)變的臃腫不堪,代碼變得難以閱讀、難以維護(hù)。這個(gè)時(shí)候,如果我們用責(zé)任鏈模式來重寫校驗(yàn)的功能系統(tǒng),那么就會(huì)像下面一樣:
通過責(zé)任鏈模式,多個(gè)處理者都有機(jī)會(huì)處理請(qǐng)求。將原本的模塊分成多個(gè)處理者來處理,符合單一職責(zé)原則,代碼的可讀性也大大提高。另外,非常容易擴(kuò)展,需要新的校驗(yàn)功能時(shí),只要添加新的處理者即可,符合開閉原則。
責(zé)任鏈模式結(jié)構(gòu)
下面我們來完成一個(gè)責(zé)任鏈模式結(jié)構(gòu)的代碼。每個(gè)處理者的功能非常簡(jiǎn)單,就是處理請(qǐng)求,然后設(shè)置下一個(gè)請(qǐng)求處理者。下面是示例代碼:
abstract class AHandler { protected $nextHandler = null; public function setNext (AHandler $handler) { $this->nextHandler = $handler; } abstract public function handler (); } class Handler1 extends AHandler { public function handler() { $handled = false; // 處理請(qǐng)求 echo 'Handler1接受到了請(qǐng)求' . PHP_EOL; if (!$handled && $this->nextHandler) { $this->nextHandler->handler(); } } } class Handler2 extends AHandler { public function handler() { $handled = false; // 處理請(qǐng)求 echo 'Handler2接受到了請(qǐng)求' . PHP_EOL; if (!$handled && $this->nextHandler) { $this->nextHandler->handler(); } } } class Handler3 extends AHandler { public function handler() { $handled = false; // 處理請(qǐng)求 echo 'Handler3接受到了請(qǐng)求' . PHP_EOL; if (!$handled && $this->nextHandler) { $this->nextHandler->handler(); } } }
使用代碼實(shí)例如下:
$handler1 = new Handler1(); $handler2 = new Handler2(); $handler3 = new Handler3(); $handler1->setNext($handler2); $handler2->setNext($handler3); $handler1->handler();
上述代碼雖然完成了責(zé)任鏈的結(jié)構(gòu),但還有一些問題,如果程序員對(duì)業(yè)務(wù)或責(zé)任鏈模式不太清楚的話,可能會(huì)忘記在handler方法里加入下面這段代碼:
if (!$handled && $this->nextHandler) { $this->nextHandler->handler(); }
那么就會(huì)造成責(zé)任鏈中斷。另外,如果我們的處理者有10個(gè)或者更多,那么就要new出10個(gè)處理者,然后還要執(zhí)行9次setNext。如果一不小心,寫錯(cuò)了,就尷尬了。
現(xiàn)在,我們修改下上述的代碼,修改后的代碼如下:
abstract class AHandler { protected $nextHandler = null; public function setNext (AHandler $handler) { $this->nextHandler = $handler; } // 使用了模板方法模式,防止程序員忘記寫下段代碼 public function handler () { if (!$this->doHandler() && $this->nextHandler) { $this->nextHandler->handler(); } } abstract public function doHandler (); } class Handler1 extends AHandler { public function doHandler() { $handled = false; // 處理請(qǐng)求 echo 'Handler1接受到了請(qǐng)求' . PHP_EOL; return $handled; } } class Handler2 extends AHandler { public function doHandler() { $handled = false; // 處理請(qǐng)求 echo 'Handler2接受到了請(qǐng)求' . PHP_EOL; return $handled; } } class Handler3 extends AHandler { public function doHandler() { $handled = false; // 處理請(qǐng)求 echo 'Handler3接受到了請(qǐng)求' . PHP_EOL; return $handled; } } class HandlerChain { private $handlerChains = []; public function __construct(array $handlerChains) { $this->handlerChains = $handlerChains; } public function addHandler (AHandler $handler) { $this->handlerChains[] = $handler; } public function handler () { $hdCnt = count($this->handlerChains); for ($i = 0; $i < $hdCnt; $i ++) { if (isset($this->handlerChains[$i]) && isset($this->handlerChains[$i+1])) { $this->handlerChains[$i]->setNext($this->handlerChains[$i+1]); } } $this->handlerChains[0]->handler(); } }
然后,使用代碼如下:
$handler1 = new Handler1(); $handler2 = new Handler2(); $handler3 = new Handler3(); $handerChian = new HandlerChain([$handler1, $handler2, $handler3]); $handerChian->handler();
更簡(jiǎn)單的實(shí)現(xiàn)方法
其實(shí)有一種更加簡(jiǎn)單的實(shí)現(xiàn)責(zé)任鏈模式的方法,代碼如下:
abstract class AHandler { abstract public function handler (); } class Handler1 extends AHandler { public function handler() { $handled = false; // 處理請(qǐng)求 echo 'Handler1接受到了請(qǐng)求' . PHP_EOL; return $handled; } } // Handler2、Handler3代碼省略 class HandlerChain { private $handlerChains = []; public function __construct(array $handlerChains) { $this->handlerChains = $handlerChains; } public function addHandler (AHandler $handler) { $this->handlerChains[] = $handler; } public function handler () { foreach ($this->handlerChains as $handler) { if ($handler->handler()) { break; } } } }
總結(jié)
通過責(zé)任鏈模式,多個(gè)處理者都有機(jī)會(huì)處理請(qǐng)求。將原本的模塊分成多個(gè)處理者來處理,符合單一職責(zé)原則,代碼的可讀性也大大提高。另外,非常容易擴(kuò)展,需要新的功能時(shí),只要添加新的處理者即可。
一般設(shè)計(jì)模式的定義是,處理者如果不能夠處理該請(qǐng)求,就將請(qǐng)求傳遞給后一個(gè)處理者。其實(shí),他也有一個(gè)變體,就是每個(gè)處理者都會(huì)處理請(qǐng)求。
上述就是小編為大家分享的什么是php中責(zé)任鏈模式了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。