PHP設(shè)計(jì)模式之觀察者模式觀察者,貌似在很多科幻作品中都會(huì)有這個(gè)角色的出現(xiàn)。比如我很喜歡的一部美劇《危機(jī)邊緣》,在這個(gè)劇集中,觀察者不停的穿越時(shí)空記錄著各種各樣的人或事。但是,設(shè)計(jì)模式中的觀察者可不只是站在邊上看哦,這里的觀察者是針對(duì)主體發(fā)生的狀態(tài)改變來做出對(duì)應(yīng)的動(dòng)作。 Gof類圖及解釋GoF定義:定義對(duì)象間的一種一對(duì)多的依賴關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí),所有依賴于它的對(duì)象都得到通知并被自動(dòng)更新 GoF類圖
 代碼實(shí)現(xiàn)
interface Observer { public function update(Subject $subject): void; }
觀察者的抽象接口,沒啥可說的吧,就是讓你實(shí)現(xiàn)一個(gè)具體的Update就可以了 class ConcreteObserver implements Observer { private $observerState = ''; function update(Subject $subject): void { $this->observerState = $subject->getState(); echo '執(zhí)行觀察者操作!當(dāng)前狀態(tài):' . $this->observerState; } }
具體的觀察者,實(shí)現(xiàn)update()方法,這里我們拿到了Subject類,從而可以獲得其中的狀態(tài) class Subject { private $observers = []; private $stateNow = ''; public function attach(Observer $observer): void { array_push($this->observers, $observer); } public function detach(Observer $observer): void { $position = 0; foreach ($this->observers as $ob) { if ($ob == $observer) { array_splice($this->observers, ($position), 1); } ++$position; } } public function notify(): void { foreach ($this->observers as $ob) { $ob->update($this); } } }
Subject父類,維護(hù)一個(gè)觀察者數(shù)組,然后有添加、刪除以及循環(huán)遍歷這個(gè)數(shù)組的方法,目的是能夠方便的管理所有的觀察者 class ConcreteSubject extends Subject{ public function setState($state) { $this->stateNow = $state; $this->notify(); }
public function getState() { return $this->stateNow; } }
Subject的實(shí)現(xiàn)類,只是更新了狀態(tài),在這個(gè)狀態(tài)發(fā)生改變的時(shí)候,調(diào)用觀察者遍歷的方法進(jìn)行所有觀察的update()操作 觀察者,其實(shí)就是自身做了一個(gè)更新(update),而Subject,可以批量的執(zhí)行觀察者,請(qǐng)注意,我們不需要去修改目標(biāo)類中的任何代碼,只需要從外部添加就可以了,所以就讓目標(biāo)和觀察者解耦互相之間不用關(guān)心對(duì)方的情況了 觀察者可以記錄目標(biāo)的狀態(tài),也可以不用記錄,比如我們發(fā)完短信后的數(shù)據(jù)庫更新或者插入操作,只有短信接口發(fā)送成功后我們?cè)傩薷亩绦艛?shù)據(jù)的狀態(tài)就可以了,不一定完全需要將目標(biāo)的發(fā)送狀態(tài)傳送給觀察者 當(dāng)一個(gè)類在發(fā)生改變時(shí),不知道可能會(huì)對(duì)其他多少類產(chǎn)生影響,這個(gè)時(shí)候觀察者非常有用 觀察者模式中還是存在著耦合,那就是目標(biāo)類中有一個(gè)觀察者對(duì)象列表,如果觀察者沒有實(shí)現(xiàn)update()方法,那么就會(huì)出現(xiàn)問題
接著拿我們的手機(jī)工廠說事兒,這次好嘛,被一幫山寨機(jī)盯上了(觀察者),我出什么功能(狀態(tài)更新),他們就對(duì)應(yīng)的出一樣的功能(更新),而且還在我的基礎(chǔ)上做了更多的東西,美其名曰:微創(chuàng)新!你說氣人不氣人。好吧,我也派出了一幫市場(chǎng)調(diào)查人員(觀察者),去幫我觀察別人家的手機(jī)都出了什么功能(狀態(tài)更新),然后我們也照搬過來搞點(diǎn)微創(chuàng)新,大家共同進(jìn)步嘛?。?/em> 完整代碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/observer.php 實(shí)例這次我們從訂單說起,不過還是有短信發(fā)送的事兒。當(dāng)一般的電商平臺(tái)有人下單之后,需要做的事情非常多,比如修改庫存、發(fā)送短信或者推送告訴商家有人下單了,告訴買家下單成功了,支付成功了??傊褪且患虑榈陌l(fā)生會(huì)導(dǎo)致各種事件的產(chǎn)生。其實(shí),這里就引出了另一個(gè)非常出名的模式訂閱發(fā)布模式。這個(gè)模式可以說是觀察者的升級(jí)模式,這個(gè)系列的文章不會(huì)細(xì)講,但是大家可以去看看Laravel中的發(fā)布訂閱及事件監(jiān)聽方面的內(nèi)容。 訂單售出類圖
 完整源碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/order-observer.php interface Observer { public function update($obj); }
class Message implements Observer { //....
function update($obj) { echo '發(fā)送新訂單短信(' . $obj->mobile . ')通知給商家!'; }
//.... }
class Goods implements Observer { //....
public function update($obj) { echo '修改商品' . $obj->goodsId . '的庫存!'; }
//.... }
class Order { private $observers = []; public function attach($ob) { $this->observers[] = $ob; }
public function detach($ob) { $position = 0; foreach ($this->observers as $ob) { if ($ob == $observer) { array_splice($this->observers, ($position), 1); } ++$position; } } public function notify($obj) { foreach ($this->observers as $ob) { $ob->update($obj); } } public function sale() { // 商品賣掉了 // .... $obj = new stdClass(); $obj->mobile = '13888888888'; $obj->goodsId = 'Order11111111'; $this->notify($obj); } }
$message = new Message(); $goods = new Goods(); $order = new Order(); $order->attach($message); $order->attach($goods);
// 訂單賣出了??! $order->sale();
說明
我們沒有完全的遵守GoF類圖,雖說GoF是圣經(jīng),但也并不是我們必須要完全遵守的,我們可以針對(duì)具體的業(yè)務(wù)情況進(jìn)行合適的裁剪使用 訂單狀態(tài)通過sale()方法產(chǎn)生變化后,直接調(diào)用notify方法進(jìn)行觀察者的調(diào)用 發(fā)短信、發(fā)推送都可以拆開由一個(gè)一個(gè)的觀察者來實(shí)現(xiàn),這些觀察者不一定只有這一個(gè)方法,但只要實(shí)現(xiàn)共同的接口就可以了 商品庫存和消息發(fā)送其實(shí)就是兩個(gè)本身完全不沾邊的類,但它們只需要實(shí)現(xiàn)一樣的接口就好啦 PHP的SPL擴(kuò)展中已經(jīng)為我們準(zhǔn)備好了一套觀察者接口,大家可以試試哦,使用原生支持的觀察者模式能省不少事兒呢!
SPL擴(kuò)展實(shí)現(xiàn)觀察者模式-完整源碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/spl_observer.php 下期看點(diǎn)循環(huán)是編程語言的一個(gè)亮點(diǎn),因?yàn)檫@個(gè)能力讓編程語言做出來的軟件可以替代人們?nèi)プ龊芏嘀貜?fù)的勞動(dòng)。一說到這里,有的人馬上就會(huì)想到,莫非我們下次講的就是迭代器模式?拭目以待吧!
|