PHP设计模式之观察者模式(Observer)详细介绍和代码实例


Posted in PHP onApril 08, 2014

【意图】

定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新【GOF95】 又称为发布-订阅(Publish-Subscribe)模式、模型-视图(Model-View)模式、源-监听(Source-Listener)模式、或从属者(Dependents)模式

【观察者模式结构图】

PHP设计模式之观察者模式(Observer)详细介绍和代码实例

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

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)
    {
        //调用邮件接口,发送邮件
    }
}
?>
PHP 相关文章推荐
复杂检索数据并分页显示的处理方法
Oct 09 PHP
php中变量及部分适用方法
Mar 27 PHP
php 无限级分类学习参考之对ecshop无限级分类的解析 带详细注释
Mar 23 PHP
ThinkPHP提示错误Fatal error: Allowed memory size的解决方法
Feb 12 PHP
php读取文件内容到数组的方法
Mar 16 PHP
PHP中余数、取余的妙用
Jun 29 PHP
Yii框架表单模型和验证用法
May 20 PHP
PHP socket 模拟POST 请求实例代码
Jul 18 PHP
全面解析PHP面向对象的三大特征
Jun 10 PHP
浅谈PHP中new self()和new static()的区别
Aug 11 PHP
PHP mongodb操作类定义与用法示例【适合mongodb2.x和mongodb3.x】
Jun 16 PHP
Yii框架getter与setter方法功能与用法分析
Oct 22 PHP
关于PHP的curl开启问题探讨
Apr 08 #PHP
PHP中Session引起的脚本阻塞问题解决办法
Apr 08 #PHP
PHP中比较两个字符串找出第一个不同字符位置例子
Apr 08 #PHP
PHP用星号隐藏部份用户名、身份证、IP、手机号等实例
Apr 08 #PHP
php实现telnet功能示例
Apr 08 #PHP
C#使用PHP服务端的Web Service通信实例
Apr 08 #PHP
php实现水仙花数的4个示例分享
Apr 08 #PHP
You might like
用PHP制作静态网站的模板框架(三)
2006/10/09 PHP
php上传文件的增强函数
2010/07/21 PHP
php打造智能化的柱状图程序,用于报表等
2015/06/19 PHP
Symfony2框架学习笔记之HTTP Cache用法详解
2016/03/18 PHP
PHP实现求两个字符串最长公共子串的方法示例
2017/11/17 PHP
php实现二叉树中和为某一值的路径方法
2018/10/14 PHP
jquery easyui中treegrid用法的简单实例
2014/02/18 Javascript
js document.write()使用介绍
2014/02/21 Javascript
JavaScript中Cookie操作实例
2015/01/09 Javascript
kindeditor编辑器点中图片滚动条往上顶的bug
2015/07/05 Javascript
javascript时间戳和日期字符串相互转换代码(超简单)
2016/06/22 Javascript
JS与HTML结合实现流程进度展示条思路详解
2017/09/03 Javascript
jQuery实现的文字逐行向上间歇滚动效果示例
2017/09/06 jQuery
vue 解决循环引用组件报错的问题
2018/09/06 Javascript
webpack4.x CommonJS模块化浅析
2018/11/09 Javascript
JavaScript创建防篡改对象的方法分析
2018/12/30 Javascript
通过JQuery,JQueryUI和Jsplumb实现拖拽模块
2019/06/18 jQuery
python中stdout输出不缓存的设置方法
2014/05/29 Python
python之virtualenv的简单使用方法(必看篇)
2017/11/25 Python
wxPython+Matplotlib绘制折线图表
2019/11/19 Python
Django 设置多环境配置文件载入问题
2020/02/25 Python
HTML5通用接口详解
2016/06/12 HTML / CSS
阿拉伯世界最大的电子卖场:Souq埃及
2016/08/01 全球购物
Melijoe时尚童装德国官网:Melijoe德国
2016/09/03 全球购物
Coccinelle官网:意大利的著名皮具品牌
2019/05/15 全球购物
AJax面试题
2014/11/25 面试题
介绍一下mysql的日期和时间函数
2013/03/28 面试题
工商管理专业学生的自我评价
2013/10/01 职场文书
思想汇报范文
2013/11/04 职场文书
学生打架检讨书1000字
2014/01/16 职场文书
幼儿园优秀教师事迹
2014/02/13 职场文书
2014年党务公开方案
2014/05/08 职场文书
党员群众路线对照检查材料思想汇报
2014/09/17 职场文书
学习走群众路线心得体会
2014/11/05 职场文书
公司2014年度工作总结
2014/12/10 职场文书
淮阳太昊陵导游词
2015/02/10 职场文书