定義對象間的一種一對多的依賴關係,當一個對象的狀態發生改變時,所有依賴於它的對象都得到通知並被自動更新【GOF95】 又稱為發布-訂閱(Publish-Subscribe)模式、模型-視圖(Model-View)模式、源-監聽(Source-Listener)模式、或從屬者(Dependents)模式
<?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)
{
//調用郵件介面,發送郵件
}
}
?>