設計模式之觀察者模式

<?php
//1.有一個主題接口,提供觀察者註冊和通知的
//2.有一堆觀察者,觀察者先向主題註冊,然後主題通知觀察者幹什麼
//主題接口
interface Subject {
	public function register(Observer $observer);
	public function notify();
}
//觀察者接口
interface Observer {
	public function watch();
}
class Action implements Subject {
	public $observers = array ();
	public function register(Observer $observer) {
		$this->observers [] = $observer;
	}
	public function notify() {
		foreach ( $this->observers as $observer ) {
			$observer->watch ();
		}
	}
}
class Cat implements Observer {
	public function watch() {
		echo 'cat watch tv';
		echo '<br/>';
	}
}
class Person implements Observer {
	public function watch() {
		echo 'person watch tv';
		echo '<br/>';
	}
}

class Dog implements Observer {
	public function watch() {
		echo 'dog watch tv';
		echo '<br/>';
	}
}
$action = new Action ();
$action->register ( new Cat () );
$action->register ( new Dog () );
$action->register ( new Person () );
$action->notify ();

//php給出的主題接口和觀察者接口 SplSubject SplObserver
class test implements SplSubject {
	public $observers = array ();
	public function attach(SplObserver $observer) {
		$this->observers [] = $observer;
	}
	public function detach(SplObserver $observer) {
		$index = array_search ( $observer, $this->observers );
		if ($index) {
			unset ( $this->observers [$index] );
		}
	}
	public function notify() {
		foreach ( $this->observers as $observer ) {
			$observer->update ();
		}
	}

}
class test2 implements SplObserver {
	public function update(SplSubject $subject) {
		echo $subject->mall;
	}
}

/**
 * 觀察者模式應用場景實例
 *
 * 免責聲明:本文只是以哈票網舉例,示例中並未涉及哈票網任何業務代碼,全部原創,如有雷同,純屬巧合。
 *
 * 場景描述:
 * 哈票以購票爲核心業務(此模式不限於該業務),但圍繞購票會產生不同的其他邏輯,如:
 * 1、購票後記錄文本日誌
 * 2、購票後記錄數據庫日誌
 * 3、購票後發送短信
 * 4、購票送抵扣卷、兌換卷、積分
 * 5、其他各類活動等
 *
 * 傳統解決方案:
 * 在購票邏輯等類內部增加相關代碼,完成各種邏輯。
 *
 * 存在問題:
 * 1、一旦某個業務邏輯發生改變,如購票業務中增加其他業務邏輯,需要修改購票核心文件、甚至購票流程。
 * 2、日積月累後,文件冗長,導致後續維護困難。
 *
 * 存在問題原因主要是程序的"緊密耦合",使用觀察模式將目前的業務邏輯優化成"鬆耦合",達到易維護、易修改的目的,
 * 同時也符合面向接口編程的思想。
 *
 * 觀察者模式典型實現方式:
 * 1、定義2個接口:觀察者(通知)接口、被觀察者(主題)接口
 * 2、定義2個類,觀察者對象實現觀察者接口、主題類實現被觀者接口
 * 3、主題類註冊自己需要通知的觀察者
 * 4、主題類某個業務邏輯發生時通知觀察者對象,每個觀察者執行自己的業務邏輯。
 *
 * 示例:如以下代碼
 *
 */
date_default_timezone_set ( 'PRC' ); //設置中國時區
#===================定義觀察者、被觀察者接口============
/**
 *
 * 觀察者接口(通知接口)
 *
 */
interface ITicketObserver //觀察者接口
{
	function onBuyTicketOver($sender, $args); //得到通知後調用的方法
}
/**
 *
 * 主題接口
 *
 */
interface ITicketObservable //被觀察對象接口
{
	function addObserver($observer); //提供註冊觀察者方法
}
#====================主題類實現========================
/**
 *
 * 主題類(購票)
 *
 */
class HipiaoBuy implements ITicketObservable { //實現主題接口(被觀察者)
	private $_observers = array (); //通知數組(觀察者)
	public function buyTicket($ticket) //購票核心類,處理購票流程
{
		// TODO 購票邏輯
		//循環通知,調用其onBuyTicketOver實現不同業務邏輯
		foreach ( $this->_observers as $obs )
			$obs->onBuyTicketOver ( $this, $ticket ); //$this 可用來獲取主題類句柄,在通知中使用
	}
	//添加通知
	public function addObserver($observer) //添加N個通知
{
		$this->_observers [] = $observer;
	}
}
#=========================定義多個通知====================
//短信日誌通知
class HipiaoMSM implements ITicketObserver {
	public function onBuyTicketOver($sender, $ticket) {
		echo (date ( 'Y-m-d H:i:s' ) . " 短信日誌記錄:購票成功:$ticket<br>");
	}
}
//文本日誌通知
class HipiaoTxt implements ITicketObserver {
	public function onBuyTicketOver($sender, $ticket) {
		echo (date ( 'Y-m-d H:i:s' ) . " 文本日誌記錄:購票成功:$ticket<br>");
	}
}
//抵扣卷贈送通知
class HipiaoDiKou implements ITicketObserver {
	public function onBuyTicketOver($sender, $ticket) {
		echo (date ( 'Y-m-d H:i:s' ) . " 贈送抵扣卷:購票成功:$ticket 贈送10元抵扣卷1張。<br>");
	}
}
#============================用戶購票====================
$buy = new HipiaoBuy ();
$buy->addObserver ( new HipiaoMSM () ); //根據不同業務邏輯加入各種通知
$buy->addObserver ( new HipiaoTxt () );
$buy->addObserver ( new HipiaoDiKou () );
//購票
$buy->buyTicket ( "一排一號" );






/**  
  * 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 $email='[email protected]';
	public $title='測試觀察者模式';
	public $content='觀察者模式測試成功';
    //註冊觀察者
    public $observers = array();
    public function __construct($data){
    $this->email=$data['email'];
    $this->title=$data['title'];
    $this->content=$data['content'];
    }
    /**
     * 追加觀察者
     * @param SplObserver $observer 觀察者
     * @param int $type 觀察類型
     */
    public function attach(SplObserver $observer)
    {
        $this->observers[] = $observer;
    }
    /**
     * 去除觀察者
     * @param SplObserver $observer 觀察者
     * @param int $type 觀察類型
     */
    public function detach(SplObserver $observer)
    {
        if($idx = array_search($observer, $this->observers, true))
        {
            unset($this->observers[$idx]);
        }
    }
    /**
     * 滿足條件時通知觀察者
     * @param int $type 觀察類型
     */
    public function notify()
    {
        if(!empty($this->observers))
        {
            foreach($this->observers as $observer)
            {
                $observer->update($this);
            }
        }
    }
    /**
     * 添加用戶
     * @param str $username 用戶名
     * @param str $password 密碼
     * @param str $email 郵箱
     * @return bool
     */
    public function addUser()
    {
        //執行sql
        //數據庫插入成功
        $res = true;
        //調用通知觀察者
        $this->notify();
        return $res;
    }
    /**
     * 用戶信息編輯
     * @param str $username 用戶名
     * @param str $password 密碼
     * @param str $email 郵箱
     * @return bool
     */
    public function editUser()
    {
        //執行sql
        //數據庫更新成功
        $res = true;
        //調用通知觀察者
        $this->notify();
        return $res;
    }
}
/**
* 觀察者-發送郵件
*/
class Send_Mail implements SplObserver
 {
    /**
     * 相應被觀察者的變更信息
     * @param SplSubject $subject
     */
    public function update(SplSubject $subject)
    {
        $this->sendEmail($subject->email, $subject->title, $subject->content);
    }
    /**
     *發送郵件
     *@param str $email 郵箱地址
     *@param str $title 郵件標題
     *@param str $content 郵件內容
     */
    public function sendEmail($email, $title, $content)
    {
        //調用郵件接口,發送郵件
        echo "email:".$email.";title:".$title.";content:".$content;
    }
}
$data=array('email'=>'[email protected]','title'=>'註冊','content'=>'恭喜註冊成功!');
$user=new User($data);
$user->attach(new Send_Mail());
$user->addUser();

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章