簡介 觀察者模式是行為型模式的一種,定義了對象間一對多的關係。當對象的狀態發生變化時候,依賴於它的對象會得到通知。 適用場景 類似觸發鉤子事件,可做消息通知、框架底層監聽。 一個對象的改變會導致一個或多個對象發生改變,方便擴展的寫法。 優點 方便擴展,降低耦合,統一觸發規則。當需要新增或者刪除一個觀 ...
簡介
觀察者模式是行為型模式的一種,定義了對象間一對多的關係。當對象的狀態發生變化時候,依賴於它的對象會得到通知。
適用場景
- 類似觸發鉤子事件,可做消息通知、框架底層監聽。
- 一個對象的改變會導致一個或多個對象發生改變,方便擴展的寫法。
優點
方便擴展,降低耦合,統一觸發規則。當需要新增或者刪除一個觀察者的時候,只需要增加觀察者就行。
缺點
- 相比於不用觀察者而是直接依賴某些類,增加代碼的複雜度。
- 如果觀察者者被觀察者互相依賴,有產生死迴圈的可能。
補充
- 需要理清楚觀察者和被觀察者是誰,觀察者可以理解為被動受到通知的對象。被觀察者是主動發送通知的對象。
- 固定的套路,被觀察者至少需要一個添加觀察者的方法和一個通知觀察者的方法用來確定身份和發送通知(一般有三個,多一個刪除觀察者的方法),觀察者至少需要一個更新的方法用於接收被觀察者的通知。
代碼(自定義實現)
//假設用戶成功購買商品後需要發送郵件和簡訊通知
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);
}
}