簡介
觀察者模式是行為型模式的一種,定義了物件間一對多的關系,當物件的狀態發生變化時候,依賴于它的物件會得到通知,
適用場景
- 類似觸發鉤子事件,可做訊息通知、框架底層監聽,
- 一個物件的改變會導致一個或多個物件發生改變,方便擴展的寫法,
優點
方便擴展,降低耦合,統一觸發規則,當需要新增或者洗掉一個觀察者的時候,只需要增加觀察者就行,
缺點
- 相比于不用觀察者而是直接依賴某些類,增加代碼的復雜度,
- 如果觀察者者被觀察者互相依賴,有產生死回圈的可能,
補充
- 需要理清楚觀察者和被觀察者是誰,觀察者可以理解為被動受到通知的物件,被觀察者是主動發送通知的物件,
- 固定的套路,被觀察者至少需要一個添加觀察者的方法和一個通知觀察者的方法用來確定身份和發送通知(一般有三個,多一個洗掉觀察者的方法),觀察者至少需要一個更新的方法用于接收被觀察者的通知,
代碼(自定義實作)
//假設用戶成功購買商品后需要發送郵件和短信通知
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);
}
}
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/522896.html
標籤:其他
上一篇:Spring Boot 學習筆記
下一篇:BLOG-2
