首頁  >  文章  >  後端開發  >  深入淺析PHP中的觀察者模式

深入淺析PHP中的觀察者模式

青灯夜游
青灯夜游轉載
2021-06-30 19:43:243278瀏覽

在之前的文章《一文了解PHP中的適配器模式》中我們介紹了PHP中的適配器模式,以下這篇文章帶大家了解一下PHP中的觀察者模式。

深入淺析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擴展中已經為我們準備好了一套觀察者接口,大家可以試試哦,使用原生支持的觀察者模式能省不少事兒呢!

SPL擴充實作觀察者模式-完整原始碼:https://github.com/zhangyue0503/designpatterns-php/blob/master/06.observer/source/spl_observer.php

本文轉載自:https://juejin.cn/post/6844903930262978574

作者:硬核心專案經理

推薦學習: 《PHP影片教學

以上是深入淺析PHP中的觀察者模式的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:juejin.cn。如有侵權,請聯絡admin@php.cn刪除