溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何理解PHP設(shè)計(jì)模式中的觀察者模式

發(fā)布時(shí)間:2021-10-08 10:41:09 來源:億速云 閱讀:115 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹“如何理解PHP設(shè)計(jì)模式中的觀察者模式”,在日常操作中,相信很多人在如何理解PHP設(shè)計(jì)模式中的觀察者模式問題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”如何理解PHP設(shè)計(jì)模式中的觀察者模式”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!

【意圖】

定義對(duì)象間的一種一對(duì)多的依賴關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí),所有依賴于它的對(duì)象都得到通知并被自動(dòng)更新【GOF95】 又稱為發(fā)布-訂閱(Publish-Subscribe)模式、模型-視圖(Model-View)模式、源-監(jiān)聽(Source-Listener)模式、或從屬者(Dependents)模式

【觀察者模式結(jié)構(gòu)圖】

如何理解PHP設(shè)計(jì)模式中的觀察者模式


【觀察者模式中主要角色】

1.抽象主題(Subject)角色:主題角色將所有對(duì)觀察者對(duì)象的引用保存在一個(gè)集合中,每個(gè)主題可以有任意多個(gè)觀察者。 抽象主題提供了增加和刪除觀察者對(duì)象的接口。
2.抽象觀察者(Observer)角色:為所有的具體觀察者定義一個(gè)接口,在觀察的主題發(fā)生改變時(shí)更新自己。
3.具體主題(ConcreteSubject)角色:存儲(chǔ)相關(guān)狀態(tài)到具體觀察者對(duì)象,當(dāng)具體主題的內(nèi)部狀態(tài)改變時(shí),給所有登記過的觀察者發(fā)出通知。具體主題角色通常用一個(gè)具體子類實(shí)現(xiàn)。
4.具體觀察者(ConcretedObserver)角色:存儲(chǔ)一個(gè)具體主題對(duì)象,存儲(chǔ)相關(guān)狀態(tài),實(shí)現(xiàn)抽象觀察者角色所要求的更新接口,以使得其自身狀態(tài)和主題的狀態(tài)保持一致。

【觀察者模式的優(yōu)點(diǎn)和缺點(diǎn)】

觀察者模式的優(yōu)點(diǎn):

1.觀察者和主題之間的耦合度較小;
2.支持廣播通信;

觀察者模式的缺點(diǎn):

由于觀察者并不知道其它觀察者的存在,它可能對(duì)改變目標(biāo)的最終代價(jià)一無所知。這可能會(huì)引起意外的更新。


【觀察者模式適用場(chǎng)景】

當(dāng)一個(gè)抽象模型有兩個(gè)方面,其中一個(gè)方面依賴于另一個(gè)方面。
當(dāng)對(duì)一個(gè)對(duì)象的改變需要同時(shí)改變其它對(duì)象,而不知道具體有多少個(gè)對(duì)象待改變。
當(dāng)一個(gè)對(duì)象必須通知其它對(duì)象,而它又不能假定其它對(duì)象是誰。換句話說,你不希望這些對(duì)象是緊密耦合的。

【觀察者模式與其它模式】

1.中介者模式(Mediator):通過封裝復(fù)雜的更新語義,ChangeManager充當(dāng)目標(biāo)和觀察者之間的中介者。
2.單例模式(singleton模式):ChangeManager可使用Singleton模式來保證它是唯一的并且是可全局訪問的。


【觀察者模式PHP示例】

復(fù)制代碼 代碼如下:


<?php

/**
* 觀察者模式
* @package design pattern
*/

/**
* 抽象主題角色
*/
interface Subject {

    /**
     * 增加一個(gè)新的觀察者對(duì)象
     * @param Observer $observer
     */
    public function attach(Observer $observer);

    /**
     * 刪除一個(gè)已注冊(cè)過的觀察者對(duì)象
     * @param Observer $observer
     */
    public function detach(Observer $observer);

    /**
     * 通知所有注冊(cè)過的觀察者對(duì)象
     */
    public function notifyObservers();
}

/**
* 具體主題角色
*/
class ConcreteSubject implements Subject {

    private $_observers;

    public function __construct() {
        $this->_observers = array();
    }

    /**
     * 增加一個(gè)新的觀察者對(duì)象
     * @param Observer $observer
     */
    public function attach(Observer $observer) {
        return array_push($this->_observers, $observer);
    }

    /**
     * 刪除一個(gè)已注冊(cè)過的觀察者對(duì)象
     * @param Observer $observer
     */
    public function detach(Observer $observer) {
        $index = array_search($observer, $this->_observers);
        if ($index === FALSE || ! array_key_exists($index, $this->_observers)) {
            return FALSE;
        }

        unset($this->_observers[$index]);
        return TRUE;
    }

    /**
     * 通知所有注冊(cè)過的觀察者對(duì)象
     */
    public function notifyObservers() {
        if (!is_array($this->_observers)) {
            return FALSE;
        }

        foreach ($this->_observers as $observer) {
            $observer->update();
        }

        return TRUE;
    }

}

/**
* 抽象觀察者角色
*/
interface Observer {

    /**
     * 更新方法
     */
    public function update();
}

class ConcreteObserver implements Observer {

    /**
     * 觀察者的名稱
     * @var <type>
     */
    private $_name;

    public function __construct($name) {
        $this->_name = $name;
    }

    /**
     * 更新方法
     */
    public function update() {
        echo 'Observer', $this->_name, ' has notified.<br />';
    }

}
實(shí)例化類:
$subject = new ConcreteSubject();

/* 添加第一個(gè)觀察者 */
$observer1 = new ConcreteObserver('Martin');
$subject->attach($observer1);

echo '<br /> The First notify:<br />';
$subject->notifyObservers();

/* 添加第二個(gè)觀察者 */
$observer2 = new ConcreteObserver('phppan');
$subject->attach($observer2);

echo '<br /> The Second notify:<br />';
$subject->notifyObservers();

/* 刪除第一個(gè)觀察者 */
$subject->detach($observer1);

echo '<br /> The Third notify:<br />';
$subject->notifyObservers();
具體案例:

<?php
 /** 
  * 3.1php設(shè)計(jì)模式-觀測(cè)者模式 
  * 3.1.1概念:其實(shí)觀察者模式這是一種較為容易去理解的一種模式吧,它是一種事件系統(tǒng),意味 
  *          著這一模式允許某個(gè)類觀察另一個(gè)類的狀態(tài),當(dāng)被觀察的類狀態(tài)發(fā)生改變的時(shí)候, 
  *          觀察類可以收到通知并且做出相應(yīng)的動(dòng)作;觀察者模式為您提供了避免組件之間
  *          緊密耦合的另一種方法
  * 3.1.2關(guān)鍵點(diǎn):
  *        1.被觀察者->追加觀察者;->一處觀察者;->滿足條件時(shí)通知觀察者;->觀察條件
  *        2.觀察者 ->接受觀察方法
  * 3.1.3缺點(diǎn):
  * 3.1.4觀察者模式在PHP中的應(yīng)用場(chǎng)合:在web開發(fā)中觀察者應(yīng)用的方面很多
  *        典型的:用戶注冊(cè)(驗(yàn)證郵件,用戶信息激活),購物網(wǎng)站下單時(shí)郵件/短信通知等
  * 3.1.5php內(nèi)部的支持
  *        SplSubject 接口,它代表著被觀察的對(duì)象,
  *        其結(jié)構(gòu):
  *        interface SplSubject
  *        {
  *            public function attach(SplObserver $observer);
  *            public function detach(SplObserver $observer);
  *            public function notify();
  *        }
  *        SplObserver 接口,它代表著充當(dāng)觀察者的對(duì)象,
  *        其結(jié)構(gòu):
  *        interface SplObserver
  *        {  
  *            public function update(SplSubject $subject);
  *        }
  */

 /**
  * 用戶登陸-詮釋觀察者模式
  */
class User implements SplSubject {
    //注冊(cè)觀察者
    public $observers = array();

    //動(dòng)作類型
    CONST OBSERVER_TYPE_REGISTER = 1;//注冊(cè)
    CONST OBSERVER_TYPE_EDIT = 2;//編輯

    /**
     * 追加觀察者
     * @param SplObserver $observer 觀察者
     * @param int $type 觀察類型
     */
    public function attach(SplObserver $observer, $type)
    {
        $this->observers[$type][] = $observer;
    }

    /**
     * 去除觀察者
     * @param SplObserver $observer 觀察者
     * @param int $type 觀察類型
     */
    public function detach(SplObserver $observer, $type)
    {
        if($idx = array_search($observer, $this->observers[$type], true))
        {
            unset($this->observers[$type][$idx]);
        }
    }

    /**
     * 滿足條件時(shí)通知觀察者
     * @param int $type 觀察類型
     */
    public function notify($type)
    {
        if(!empty($this->observers[$type]))
        {
            foreach($this->observers[$type] as $observer)
            {
                $observer->update($this);
            }
        }
    }

    /**
     * 添加用戶
     * @param str $username 用戶名
     * @param str $password 密碼
     * @param str $email 郵箱
     * @return bool
     */
    public function addUser()
    {

        //執(zhí)行sql

        //數(shù)據(jù)庫插入成功
        $res = true;

        //調(diào)用通知觀察者
        $this->notify(self::OBSERVER_TYPE_REGISTER);

        return $res;
    }

    /**
     * 用戶信息編輯
     * @param str $username 用戶名
     * @param str $password 密碼
     * @param str $email 郵箱
     * @return bool
     */
    public function editUser()
    {

        //執(zhí)行sql

        //數(shù)據(jù)庫更新成功
        $res = true;

        //調(diào)用通知觀察者
        $this->notify(self::OBSERVER_TYPE_EDIT);

        return $res;
    }
}

/**
* 觀察者-發(fā)送郵件
*/
class Send_Mail implements SplObserver
 {
    /**
     * 相應(yīng)被觀察者的變更信息
     * @param SplSubject $subject
     */
    public function update(SplSubject $subject)
    {
        $this->sendMail($subject->email, $title, $content);
    }

    /**
     *發(fā)送郵件
     *@param str $email 郵箱地址
     *@param str $title 郵件標(biāo)題
     *@param str $content 郵件內(nèi)容
     */
    public function sendEmail($email, $title, $content)
    {
        //調(diào)用郵件接口,發(fā)送郵件
    }
}
?>

到此,關(guān)于“如何理解PHP設(shè)計(jì)模式中的觀察者模式”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI