PHP設(shè)計(jì)模式中觀察者模式講解
簡(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è)用戶成功購買商品后需要發(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); } } //購買商品,觸發(fā)通知 public function buyGoods() { //todo 訂單操作 echo '商品購買完成' . 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類庫,用于解決典型問題的一組接口與類的集合。
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 '購買成功' . 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)文章
微信公眾平臺(tái)開發(fā)關(guān)注及取消關(guān)注事件的方法
這篇文章主要介紹了微信公眾平臺(tái)開發(fā)關(guān)注及取消關(guān)注事件的方法,較為詳細(xì)的分析了微信公眾平臺(tái)設(shè)置關(guān)注的技巧,并附帶了相關(guān)參數(shù)的說明,具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2014-12-12php curl 登錄163郵箱并抓取郵箱好友列表的代碼(經(jīng)測(cè)試)
PHP模擬登陸獲取163郵箱聯(lián)系人的實(shí)現(xiàn)代碼,需要的朋友可以參考下。2011-04-04深入了解 register_globals (附register_globals=off 網(wǎng)站打不開的解決方法)
由于register_globals設(shè)置控制PHP變量訪問范圍,如果開啟會(huì)引起不必要的安全問題,所以這里對(duì)其進(jìn)行了強(qiáng)制關(guān)閉,如果站長(zhǎng)的空間不支持,可以采用以下幾種辦法進(jìn)行修改,供廣大站長(zhǎng)參考2012-06-06php多層數(shù)組與對(duì)象的轉(zhuǎn)換實(shí)例代碼
通過json_decode(json_encode($object)可以將對(duì)象一次性轉(zhuǎn)換為數(shù)組,但是object中遇到非utf-8編碼的非ascii字符則會(huì)出現(xiàn)問題,比如gbk的中文,何況json_encode和decode的性能也值得疑慮2013-08-08