首頁 > 軟體

PHP設計模式中觀察者模式詳解

2022-12-21 14:00:27

簡介

觀察者模式是行為型模式的一種,定義了物件間一對多的關係。當物件的狀態發生變化時候,依賴於它的物件會得到通知。

適用場景

類似觸發勾點事件,可做訊息通知、框架底層監聽。

一個物件的改變會導致一個或多個物件發生改變,方便擴充套件的寫法。

優點

方便擴充套件,降低耦合,統一觸發規則。當需要新增或者刪除一個觀察者的時候,只需要增加觀察者就行。

缺點

相比於不用觀察者而是直接依賴某些類,增加程式碼的複雜度。

如果觀察者者被觀察者互相依賴,有產生死迴圈的可能。

補充

需要理清楚觀察者和被觀察者是誰,觀察者可以理解為被動受到通知的物件。被觀察者是主動傳送通知的物件。

固定的套路,被觀察者至少需要一個新增觀察者的方法和一個通知觀察者的方法用來確定身份和傳送通知(一般有三個,多一個刪除觀察者的方法),觀察者至少需要一個更新的方法用於接收被觀察者的通知。

程式碼(自定義實現)

//假設使用者成功購買商品後需要傳送郵件和簡訊通知
class Order {
    private $observers = [];
    //新增觀察者
    public function attach($type, $observer) {
        $this->observers[$type] = $observer;
    }
    //對每個觀察者進行通知
    public function notify() {
        if ($this->observers == []) {
            return null;
        }
        foreach ($this->observers as $every_observer) {
            (new $every_observer)->update($this);
        }
    }
    //購買商品,觸發通知
    public function buyGoods() {
        //todo 訂單操作
        echo '商品購買完成' . PHP_EOL;
        $this->notify();
    }
}
class Mail {
    public function update($observer) {
        echo '傳送電子郵件' . PHP_EOL;
    }
}
class Sms {
    public function update($observer) {
        echo '傳送簡訊' . PHP_EOL;
    }
}
$order = new Order();
//新增觀察者
$order->attach('mail', Mail::class);
$order->attach('sms', Sms::class);
$order->buyGoods();

程式碼(基於SPL實現)

SPL(Standard PHP Library)標準PHP類庫,用於解決典型問題的一組介面與類的集合。

class OrderListener implements SplSubject {
    //觀察者列表
    public $observers;
    public function __construct() {
        //SplObjectStorage類提供從物件到資料的對映,或者通過忽略資料,提供物件集的對映。在許多需要唯一標識物件的情況下,這種雙重用途非常有用。
        $this->observers = new SplObjectStorage();
    }
    //新增要通知的物件
    public function attach(SplObserver $observer) {
        $this->observers->attach($observer);
    }
    //移除要通知的物件
    public function detach(SplObserver $observer) {
        $this->observers->detach($observer);
    }
    //通知
    public function notify() {
        //將迭代器(此處可以理解為指標)倒回到第一個儲存元素。
        $this->observers->rewind();
        //判斷指標是否有效
        while($this->observers->valid()) {
            //獲取當前的觀察者
            $curr_obj = $this->observers->current();
            //對當前觀察者進行通知
            $curr_obj->update($this);
            //向下移動指標
            $this->observers->next();
        }
    }
    //觸發通知
    public function buyGoods() {
        echo '購買成功' . PHP_EOL;
        $this->notify();
    }
}
//SplObserver介面與SplSubject介面一起使用,以實現觀察者設計模式。
class Mail implements SplObserver {
    //對被觀察的物件做相應的處理
    public function update(SplSubject $subject) {
        echo '傳送郵件' . PHP_EOL;
    }
}
class Sms implements SplObserver {
    //對被觀察的物件做相應的處理
    public function update(SplSubject $subject) {
        echo '傳送簡訊' . PHP_EOL;
    }
}
$listener = new OrderListener();
//新增觀察者
$listener->attach(new Mail());
$listener->attach(new Sms());
$listener->buyGoods();

通知程式碼(基於SPL實現的notify方法優化)

	//以上程式碼的notify方法使用原生手動調整指標的方式去實現。也可以使用foreach去遍歷實現
    public function notify() {
        foreach ($this->observers as $observer) {
            $observer->update($this);
        }
    }

到此這篇關於PHP設計模式中觀察者模式詳解的文章就介紹到這了,更多相關PHP觀察者模式內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com