PHP設(shè)計(jì)模式中觀察者模式詳解
目錄
- 簡(jiǎn)介
- 適用場(chǎng)景
- 缺點(diǎn)
- 補(bǔ)充
- 代碼(自定義實(shí)現(xiàn))
簡(jiǎn)介
觀察者模式是行為型模式的一種,定義了對(duì)象間一對(duì)多的關(guān)系。當(dāng)對(duì)象的狀態(tài)發(fā)生變化時(shí)候,依賴于它的對(duì)象會(huì)得到通知。
適用場(chǎng)景
類似觸發(fā)鉤子事件,可做消息通知、框架底層監(jiān)聽。
一個(gè)對(duì)象的改變會(huì)導(dǎo)致一個(gè)或多個(gè)對(duì)象發(fā)生改變,方便擴(kuò)展的寫法。
優(yōu)點(diǎn)
方便擴(kuò)展,降低耦合,統(tǒng)一觸發(fā)規(guī)則。當(dāng)需要新增或者刪除一個(gè)觀察者的時(shí)候,只需要增加觀察者就行。
缺點(diǎn)
相比于不用觀察者而是直接依賴某些類,增加代碼的復(fù)雜度。
如果觀察者者被觀察者互相依賴,有產(chǎn)生死循環(huán)的可能。
補(bǔ)充
需要理清楚觀察者和被觀察者是誰,觀察者可以理解為被動(dòng)受到通知的對(duì)象。被觀察者是主動(dòng)發(fā)送通知的對(duì)象。
固定的套路,被觀察者至少需要一個(gè)添加觀察者的方法和一個(gè)通知觀察者的方法用來確定身份和發(fā)送通知(一般有三個(gè),多一個(gè)刪除觀察者的方法),觀察者至少需要一個(gè)更新的方法用于接收被觀察者的通知。
代碼(自定義實(shí)現(xiàn))
//假設(shè)用戶成功購(gòu)買商品后需要發(fā)送郵件和短信通知
class Order {
private $observers = [];
//添加觀察者
public function attach($type, $observer) {
$this->observers[$type] = $observer;
}
//對(duì)每個(gè)觀察者進(jìn)行通知
public function notify() {
if ($this->observers == []) {
return null;
}
foreach ($this->observers as $every_observer) {
(new $every_observer)->update($this);
}
}
//購(gòu)買商品,觸發(fā)通知
public function buyGoods() {
//todo 訂單操作
echo "商品購(gòu)買完成" . PHP_EOL;
$this->notify();
}
}
class Mail {
public function update($observer) {
echo "發(fā)送電子郵件" . PHP_EOL;
}
}
class Sms {
public function update($observer) {
echo "發(fā)送短信" . PHP_EOL;
}
}
$order = new Order();
//添加觀察者
$order->attach("mail", Mail::class);
$order->attach("sms", Sms::class);
$order->buyGoods();
代碼(基于SPL實(shí)現(xiàn))
SPL(Standard PHP Library)標(biāo)準(zhǔn)PHP類庫(kù),用于解決典型問題的一組接口與類的集合。
class OrderListener implements \SplSubject {
//觀察者列表
public $observers;
public function __construct() {
//SplObjectStorage類提供從對(duì)象到數(shù)據(jù)的映射,或者通過忽略數(shù)據(jù),提供對(duì)象集的映射。在許多需要唯一標(biāo)識(shí)對(duì)象的情況下,這種雙重用途非常有用。
$this->observers = new \SplObjectStorage();
}
//添加要通知的對(duì)象
public function attach(\SplObserver $observer) {
$this->observers->attach($observer);
}
//移除要通知的對(duì)象
public function detach(\SplObserver $observer) {
$this->observers->detach($observer);
}
//通知
public function notify() {
//將迭代器(此處可以理解為指針)倒回到第一個(gè)存儲(chǔ)元素。
$this->observers->rewind();
//判斷指針是否有效
while($this->observers->valid()) {
//獲取當(dāng)前的觀察者
$curr_obj = $this->observers->current();
//對(duì)當(dāng)前觀察者進(jìn)行通知
$curr_obj->update($this);
//向下移動(dòng)指針
$this->observers->next();
}
}
//觸發(fā)通知
public function buyGoods() {
echo "購(gòu)買成功" . PHP_EOL;
$this->notify();
}
}
//SplObserver接口與SplSubject接口一起使用,以實(shí)現(xiàn)觀察者設(shè)計(jì)模式。
class Mail implements \SplObserver {
//對(duì)被觀察的對(duì)象做相應(yīng)的處理
public function update(\SplSubject $subject) {
echo "發(fā)送郵件" . PHP_EOL;
}
}
class Sms implements \SplObserver {
//對(duì)被觀察的對(duì)象做相應(yīng)的處理
public function update(\SplSubject $subject) {
echo "發(fā)送短信" . PHP_EOL;
}
}
$listener = new OrderListener();
//添加觀察者
$listener->attach(new Mail());
$listener->attach(new Sms());
$listener->buyGoods();
通知代碼(基于SPL實(shí)現(xiàn)的notify方法優(yōu)化)
//以上代碼的notify方法使用原生手動(dòng)調(diào)整指針的方式去實(shí)現(xiàn)。也可以使用foreach去遍歷實(shí)現(xiàn)
public function notify() {
foreach ($this->observers as $observer) {
$observer->update($this);
}
}
到此這篇關(guān)于PHP設(shè)計(jì)模式中觀察者模式詳解的文章就介紹到這了,更多相關(guān)PHP觀察者模式內(nèi)容請(qǐng)搜索以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持!
相關(guān)文章:
1. PHP設(shè)計(jì)模式中工廠模式深入詳解2. php設(shè)計(jì)模式之備忘模式分析【星際爭(zhēng)霸游戲案例】3. 更好的構(gòu)造開發(fā)模板 五種PHP設(shè)計(jì)模式4. php設(shè)計(jì)模式之代理模式分析【星際爭(zhēng)霸游戲案例】5. 深入分析PHP設(shè)計(jì)模式6. PHP設(shè)計(jì)模式入門之狀態(tài)模式原理與實(shí)現(xiàn)方法分析7. php設(shè)計(jì)模式之觀察者模式實(shí)例詳解【星際爭(zhēng)霸游戲案例】8. PHP設(shè)計(jì)模式之迭代器模式的使用9. PHP設(shè)計(jì)模式之解釋器模式淺析10. PHP設(shè)計(jì)模式之模板方法模式Template Method Pattern詳解

網(wǎng)公網(wǎng)安備