PHP設計模式之觀察者模式(Observer)詳細介紹和代碼執行個體_php執行個體

來源:互聯網
上載者:User

【意圖】

定義對象間的一種一對多的依賴關係,當一個對象的狀態發生改變時,所有依賴於它的對象都得到通知並被自動更新【GOF95】 又稱為發布-訂閱(Publish-Subscribe)模式、模型-視圖(Model-View)模式、源-監聽(Source-Listener)模式、或從屬者(Dependents)模式

【觀察者模式結構圖】


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

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

【觀察者模式的優點和缺點】

觀察者模式的優點:

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

觀察者模式的缺點:

由於觀察者並不知道其它觀察者的存在,它可能對改變目標的最終代價一無所知。這可能會引起意外的更新。


【觀察者模式適用情境】

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

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

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


【觀察者模式PHP樣本】

複製代碼 代碼如下:

<?php

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

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

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

    /**
     * 刪除一個登入過的觀察者對象
     * @param Observer $observer
     */
    public function detach(Observer $observer);

    /**
     * 通知所有註冊過的觀察者對象
     */
    public function notifyObservers();
}

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

    private $_observers;

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

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

    /**
     * 刪除一個登入過的觀察者對象
     * @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;
    }

    /**
     * 通知所有註冊過的觀察者對象
     */
    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 />';
    }

}
執行個體化類:
$subject = new ConcreteSubject();

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

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

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

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

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

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

<?php
 /** 
  * 3.1php設計模式-觀測者模式 
  * 3.1.1概念:其實觀察者模式這是一種較為容易去理解的一種模式吧,它是一種事件系統,意味 
  *          著這一模式允許某個類觀察另一個類的狀態,當被觀察的類狀態發生改變的時候, 
  *          觀察類可以收到通知並且做出相應的動作;觀察者模式為您提供了避免組件之間
  *          緊密耦合的另一種方法
  * 3.1.2關鍵點:
  *        1.被觀察者->追加觀察者;->一處觀察者;->滿足條件時通知觀察者;->觀察條件
  *        2.觀察者 ->接受觀察方法
  * 3.1.3缺點:
  * 3.1.4觀察者模式在PHP中的應用場合:在web開發中觀察者應用的方面很多
  *        典型的:使用者註冊(驗證郵件,使用者資訊啟用),購物網站下單時郵件/簡訊通知等
  * 3.1.5php內部的支援
  *        SplSubject 介面,它代表著被觀察的對象,
  *        其結構:
  *        interface SplSubject
  *        {
  *            public function attach(SplObserver $observer);
  *            public function detach(SplObserver $observer);
  *            public function notify();
  *        }
  *        SplObserver 介面,它代表著充當觀察者的對象,
  *        其結構:
  *        interface SplObserver
  *        {  
  *            public function update(SplSubject $subject);
  *        }
  */

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

    //動作類型
    CONST OBSERVER_TYPE_REGISTER = 1;//註冊
    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]);
        }
    }

    /**
     * 滿足條件時通知觀察者
     * @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()
    {

        //執行sql

        //資料庫插入成功
        $res = true;

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

        return $res;
    }

    /**
     * 使用者資訊編輯
     * @param str $username 使用者名稱
     * @param str $password 密碼
     * @param str $email 郵箱
     * @return bool
     */
    public function editUser()
    {

        //執行sql

        //資料庫更新成功
        $res = true;

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

        return $res;
    }
}

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

    /**
     *發送郵件
     *@param str $email 郵箱地址
     *@param str $title 郵件標題
     *@param str $content 郵件內容
     */
    public function sendEmail($email, $title, $content)
    {
        //調用郵件介面,發送郵件
    }
}
?>

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.