设计原则
在观察者模式中,会改变的是主题的状态以及观察者的数目。用这个模式,你可以改变依赖于主题状态的对象,却不必改变主题。——找出程序中会变化的方面,然后将其和固定不变的方面相分离!
主题和观察者都使用接口:观察者利用主题的接口向主题注册,而主题利用观察者接口通知观察者。这样可以让两者之间运作正常,又同时具有松耦合的优点! ——针对接口编程,不针对实现编程!
观察者模式利用“组合”将许多观察者组合进主题中。对象(观察者——主题)之间的这种关系不是通过继承产生的,而是在运行时利用组合的方式产生的。 ——多用组合,少用继承!
代码
<?php
/**
* 观察者模式
* @author: Mac
* @date: 2012/02/22
*/
class Paper{ /* 主题 */
private $_observers = array();
public function register($sub){ /* 注册观察者 */
$this->_observers[] = $sub;
}
public function trigger(){ /* 外部统一访问 */
if(!empty($this->_observers)){
foreach($this->_observers as $observer){
$observer->update();
}
}
}
}
/**
* 观察者要实现的接口
*/
interface Observerable{
public function update();
}
class Subscriber implements Observerable{
public function update(){
echo "Callback\n";
}
}
/* 测试 */
$paper = new Paper();
$paper->register(new Subscriber());
//$paper->register(new Subscriber1());
//$paper->register(new Subscriber2());
$paper->trigger();
---------------------------------------------------------------------------------------------------------
当一个对象状态发生改变后,会影响到其他几个对象的改变,这时候可以用观察者模式。像wordpress这样的应用程序中,它容外部开发组开发插件,比如用户授权的博客统计插件、积分插件,这时候可以应用观察者模式,先注册这些插件,当用户发布一篇博文后,就回自动通知相应的插件更新。
观察者模式符合接口隔离原则,实现了对象之间的松散耦合。
观察者模式UML图:
在php SPL中已经提供SplSubject和SqlOberver接口
- interface SplSubject
- {
- function attach(SplObserver $observer);
- function detach(SplObserver $observer);
- function notify();
- }
- interface SqlObserver
- {
- function update(SplSubject $subject);
- }
- class Subject implements SplSubject
- {
- private $observers;
- public function attach(SplObserver $observer)
- {
- if (!in_array($observer, $this->observers)) {
- $this->observers[] = $observer;
- }
- }
- public function detach(SplObserver $observer)
- {
- if (false != ($index = array_search($observer, $this->observers))) {
- unset($this->observers[$index]);
- }
- }
- public function post()
- {
- //post相关code
- $this->notify();
- }
- private function notify()
- {
- foreach ($this->observers as $observer) {
- $observer->update($this);
- }
- }
- public function setCount($count)
- {
- echo "数据量加" . $count;
- }
- public function setIntegral($integral)
- {
- echo "积分量加" . $integral;
- }
- }
- class Observer1 implements SplObserver
- {
- public function update($subject)
- {
- $subject-> setCount(1);
- }
- }
- class Observer2 implements SplObserver
- {
- public function update($subject)
- {
- $subject-> setIntegral(10);
- }
- }
- class Client
- {
- public function test()
- {
- $subject = new Subject();
- $subject->attach(new Observer1());
- $subject->attach(new Observer2());
- $subject->post();//输出:数据量加1 积分量加10
- }
- }