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设计模式 Composite (组合模式)
Jun 26 PHP
PHP判断文件是否存在、是否可读、目录是否存在的代码
Oct 03 PHP
浅析使用Turck-mmcache编译来加速、优化PHP代码
Jun 20 PHP
smarty内置函数config_load用法实例
Jan 22 PHP
PHP版本如何选择?应该使用哪个版本?
May 13 PHP
调用WordPress函数统计文章访问量及PHP原生计数器的实现
Mar 21 PHP
Thinkphp单字母函数使用指南
May 08 PHP
利用PHP判断是否是连乘数字串的方法示例
Jul 03 PHP
PhpStorm本地断点调试的方法步骤
May 21 PHP
实例说明js脚本语言和php脚本语言的区别
Apr 04 PHP
PHP正则表达式处理函数(PCRE 函数)实例小结
May 09 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代码质量36计
2012/09/05 PHP
php实现计算百度地图坐标之间距离的方法
2016/05/05 PHP
PHP-FPM运行状态的实时查看及监控详解
2016/11/18 PHP
PHP利用DWZ.CN服务生成短网址
2019/08/11 PHP
基于jQuery的左右滚动实现代码
2010/12/03 Javascript
jQuery 遍历-nextUntil()方法以及prevUntil()方法的使用介绍
2013/04/26 Javascript
JS 有趣的eval优化输入验证实例代码
2013/09/22 Javascript
javascript常见操作汇总
2014/09/03 Javascript
js对象的复制继承实例
2015/01/10 Javascript
JavaScript实现网页加载进度条代码超简单
2015/09/21 Javascript
JavaScript知识点总结(六)之JavaScript判断变量数据类型
2016/05/31 Javascript
JQuery Ajax 异步操作之动态添加节点功能
2017/05/24 jQuery
Jquery中.bind()、.live()、.delegate()和.on()之间的区别详解
2017/08/01 jQuery
详解ES6 CLASS在微信小程序中的应用实例
2020/04/24 Javascript
JavaScript中继承原理与用法实例入门
2020/05/09 Javascript
使用python实现拉钩网上的FizzBuzzWhizz问题示例
2014/05/05 Python
python 寻找list中最大元素对应的索引方法
2018/06/28 Python
PyCharm 配置远程python解释器和在本地修改服务器代码
2019/07/23 Python
python logging.basicConfig不生效的原因及解决
2020/02/20 Python
Python requests模块session代码实例
2020/04/14 Python
浅谈OpenCV中的新函数connectedComponentsWithStats用法
2020/07/05 Python
浅谈关于html5中图片抛物线运动的一些心得
2018/01/09 HTML / CSS
英国在线房屋中介网站:Yopa
2018/01/09 全球购物
SEPHORA丝芙兰捷克官网:购买香水、化妆品和护肤品
2018/11/26 全球购物
简述Linux文件系统通过i节点把文件的逻辑结构和物理结构转换的工作过程
2012/04/17 面试题
养殖人员的创业计划书范文
2013/12/26 职场文书
幼儿园教师培训方案
2014/02/04 职场文书
大学班级干部的自我评价分享
2014/02/10 职场文书
房产代理公证处委托书
2014/04/04 职场文书
国际贸易专业求职信
2014/06/04 职场文书
运动会方队口号
2014/06/07 职场文书
小学生教师节演讲稿
2014/09/03 职场文书
免职证明样本
2014/10/23 职场文书
运动会通讯稿600字
2015/07/20 职场文书
yyds什么意思?90后已经听不懂00后讲话了……
2022/02/03 杂记
剑指Offer之Java算法习题精讲二叉树专项训练
2022/03/21 Java/Android