PHP設計模式之觀察者模式

觀察者,貌似在很多科幻作品中都會有這個角色的出現。比如我很喜歡的一部美劇《危機邊緣》,在這個劇集中,觀察者不停的穿越時空記錄著各種各樣的人或事。但是,設計模式中的觀察者可不只是站在邊上看哦,這裡的觀察者是針對主體發生的狀態改變來做出對應的動作。

Gof類圖及解釋

GoF定義:定義物件間的一種一對多的依賴關係,當一個物件的狀態發生改變時,所有依賴於它的物件都得到通知並被自動更新

GoF類圖

PHP設計模式之觀察者模式

介面卡方法結構類圖-繼承式

程式碼實現

interface Observer{    public function update(Subject $subject): void;}

觀察者的抽象介面,沒啥可說的吧,就是讓你實現一個具體的Update就可以了

class ConcreteObserver implements Observer{    private $observerState = ‘’;    function update(Subject $subject): void    {        $this->observerState = $subject->getState();        echo ‘執行觀察者操作!當前狀態:’ 。 $this->observerState;    }}

具體的觀察者,實現update()方法,這裡我們拿到了Subject類,從而可以獲得其中的狀態

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父類,維護一個觀察者陣列,然後有新增、刪除以及迴圈遍歷這個陣列的方法,目的是能夠方便的管理所有的觀察者

class ConcreteSubject extends Subject{    public function setState($state)    {        $this->stateNow = $state;        $this->notify();    }    public function getState()    {        return $this->stateNow;    }}

Subject的實現類,只是更新了狀態,在這個狀態發生改變的時候,呼叫觀察者遍歷的方法進行所有觀察的update()操作

觀察者,其實就是自身做了一個更新(update),而Subject,可以批次的執行觀察者,請注意,我們不需要去修改目標類中的任何程式碼,只需要從外部新增就可以了,所以就讓目標和觀察者解耦互相之間不用關心對方的情況了

觀察者可以記錄目標的狀態,也可以不用記錄,比如我們發完簡訊後的資料庫更新或者插入操作,只有簡訊介面傳送成功後我們再修改簡訊資料的狀態就可以了,不一定完全需要將目標的傳送狀態傳送給觀察者

當一個類在發生改變時,不知道可能會對其他多少類產生影響,這個時候觀察者非常有用

觀察者模式中還是存在著耦合,那就是目標類中有一個觀察者物件列表,如果觀察者沒有實現update()方法,那麼就會出現問題

接著拿我們的手機工廠說事兒,這次好嘛,被一幫山寨機盯上了(觀察者),我出什麼功能(狀態更新),他們就對應的出一樣的功能(更新),而且還在我的基礎上做了更多的東西,美其名曰:微創新!你說氣人不氣人。好吧,我也派出了一幫市場調查人員(觀察者),去幫我觀察別人家的手機都出了什麼功能(狀態更新),然後我們也照搬過來搞點微創新,大家共同進步嘛!!

完整程式碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/observer.php

例項

這次我們從訂單說起,不過還是有簡訊傳送的事兒。當一般的電商平臺有人下單之後,需要做的事情非常多,比如修改庫存、傳送簡訊或者推送告訴商家有人下單了,告訴買家下單成功了,支付成功了。總之就是一件事情的發生會導致各種事件的產生。其實,這裡就引出了另一個非常出名的模式

訂閱釋出

模式。這個模式可以說是觀察者的升級模式,這個系列的文章不會細講,但是大家可以去看看Laravel中的

釋出訂閱

事件監聽

方面的內容。

訂單售出類圖

PHP設計模式之觀察者模式

訂單售出觀察者模式

完整原始碼: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 ‘傳送新訂單簡訊(’ 。 $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是聖經,但也並不是我們必須要完全遵守的,我們可以針對具體的業務情況進行合適的裁剪使用

訂單狀態透過sale()方法產生變化後,直接呼叫notify方法進行觀察者的呼叫

發簡訊、發推送都可以拆開由一個一個的觀察者來實現,這些觀察者不一定只有這一個方法,但只要實現共同的介面就可以了

商品庫存和訊息傳送其實就是兩個本身完全不沾邊的類,但它們只需要實現一樣的介面就好啦

PHP的SPL擴充套件中已經為我們準備好了一套觀察者介面,大家可以試試哦,使用原生支援的觀察者模式能省不少事兒呢!

完整原始碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/spl_observer.php

下期看點

迴圈是程式語言的一個亮點,因為這個能力讓程式語言做出來的軟體可以替代人們去做很多重複的勞動。一說到這裡,有的人馬上就會想到,莫非我們下次講的就是

迭代器

模式?拭目以待吧!