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)聽(tīng)。
一個(gè)對(duì)象的改變會(huì)導(dǎo)致一個(gè)或多個(gè)對(duì)象發(fā)生改變,方便擴(kuò)展的寫(xiě)法。
優(yōu)點(diǎn)
方便擴(kuò)展,降低耦合,統(tǒng)一觸發(fā)規(guī)則。當(dāng)需要新增或者刪除一個(gè)觀察者的時(shí)候,只需要增加觀察者就行。
缺點(diǎn)
相比于不用觀察者而是直接依賴某些類,增加代碼的復(fù)雜度。
如果觀察者者被觀察者互相依賴,有產(chǎn)生死循環(huán)的可能。
補(bǔ)充
需要理清楚觀察者和被觀察者是誰(shuí),觀察者可以理解為被動(dòng)受到通知的對(duì)象。被觀察者是主動(dòng)發(fā)送通知的對(duì)象。
固定的套路,被觀察者至少需要一個(gè)添加觀察者的方法和一個(gè)通知觀察者的方法用來(lái)確定身份和發(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ù),用于解決典型問(wèn)題的一組接口與類的集合。
class OrderListener implements \SplSubject { //觀察者列表 public $observers; public function __construct() { //SplObjectStorage類提供從對(duì)象到數(shù)據(jù)的映射,或者通過(guò)忽略數(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ì)模式入門之狀態(tài)模式原理與實(shí)現(xiàn)方法分析3. php設(shè)計(jì)模式之備忘模式分析【星際爭(zhēng)霸游戲案例】4. PHP設(shè)計(jì)模式之解釋器模式淺析5. php設(shè)計(jì)模式之正面模式實(shí)例分析【星際爭(zhēng)霸游戲案例】6. 詳解PHP設(shè)計(jì)模式之橋接模式7. PHP設(shè)計(jì)模式之模板方法模式Template Method Pattern詳解8. php設(shè)計(jì)模式之享元模式分析【星際爭(zhēng)霸游戲案例】9. PHP設(shè)計(jì)模式之中介者模式淺析10. PHP設(shè)計(jì)模式(三)建造者模式Builder實(shí)例詳解【創(chuàng)建型】
