php模式设计之 观察者模式

本文深入解析观察者模式,探讨其如何实现低耦合的通知与更新机制,通过示例展示在购票系统中的应用,有效解决业务逻辑修改导致的代码冗余问题。

观察者模式

观察者模式(Observer),当一个对象的状态发生改变时,依赖他的对象会全部收到通知,并自动更新。

场景:一个事件发生后,要执行一连串更新操作.传统的编程方式,就是在事件的代码之后直接加入处理逻辑,当更新得逻辑增多之后,代码会变得难以维护.这种方式是耦合的,侵入式的,增加新的逻辑需要改变事件主题的代码

当一个对象状态发生改变后,会影响到其他几个对象的改变,这时候可以用观察者模式。

观察者模式实现了低耦合,非侵入式的通知与更新机制

观察者模式符合接口隔离原则,实现了对象之间的松散耦合。


// 主题接口
interface Subject{
	//添加观察者
    public function register(Observer $observer);
	
	//时间通知
    public function notify();
}
// 观察者接口
interface Observer{
    public function notify();
}
// 主题
class Action implements Subject{
     public $_observers=array();
     public function register(Observer $observer){
         $this->_observers[]=$observer;
     }


	/**
     * 触发事件
     */
     public function trigger(){
         foreach ($this->_observers as $observer) {
		 	//通知观察者
             $observer->notify();
         }

     }
 }

// 观察者
class Cat implements Observer{
     public function notify(){
         echo "Cat watches TV<hr/>";
     }
 } 
 class Dog implements Observer{
     public function notify(){
         echo "Dog watches TV<hr/>";
     }
 } 
 class People implements Observer{
     public function notify(){
         echo "People watches TV<hr/>";
     }
 }



//创建一个事件
$action=new Action();
//为事件增加旁观者
$action->register(new Cat());
$action->register(new People());
$action->register(new Dog());
//执行事件 通知旁观者
$action->trigger();

实际中的应用

/**
 * 观察者模式应用场景实例
 *
 * 免责声明:本文只是以哈票网举例,示例中并未涉及哈票网任何业务代码,全部原创,如有雷同,纯属巧合。
 *
 * 场景描述:
 * 哈票以购票为核心业务(此模式不限于该业务),但围绕购票会产生不同的其他逻辑,如:
 * 1、购票后记录文本日志
 * 2、购票后记录数据库日志
 * 3、购票后发送短信
 * 4、购票送抵扣卷、兑换卷、积分
 * 5、其他各类活动等
 *
 * 传统解决方案:
 * 在购票逻辑等类内部增加相关代码,完成各种逻辑。
 *
 * 存在问题:
 * 1、一旦某个业务逻辑发生改变,如购票业务中增加其他业务逻辑,需要修改购票核心文件、甚至购票流程。
 * 2、日积月累后,文件冗长,导致后续维护困难。
 *
 * 存在问题原因主要是程序的"紧密耦合",使用观察模式将目前的业务逻辑优化成"松耦合",达到易维护、易修改的目的,
 * 同时也符合面向接口编程的思想。
 *
 * 观察者模式典型实现方式:
 * 1、定义2个接口:观察者(通知)接口、被观察者(主题)接口
 * 2、定义2个类,观察者对象实现观察者接口、主题类实现被观者接口
 * 3、主题类注册自己需要通知的观察者
 * 4、主题类某个业务逻辑发生时通知观察者对象,每个观察者执行自己的业务逻辑。
 *
 * 示例:如以下代码
 *
 */

代码如下

#===================定义观察者、被观察者接口============
/**
*
* 观察者接口(通知接口)
*
*/
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 ( "一排一号" );

运行结果如下:

2017-02-03 10:25:45 短信日志记录:购票成功:一排一号
2017-02-03 10:25:45 文本日志记录:购票成功:一排一号
2017-02-03 10:25:45 赠送抵扣卷:购票成功:一排一号 赠送10元抵扣卷1张

转载于:https://my.oschina.net/u/3683692/blog/2997106

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值