观察者模式

模式定义

观察者模式(Observer Pattern):定义对象间的一种一对多依赖关系,使得每当一个对象状态发生改变时,其相关依赖对象皆得到通知并被自动更新。观察者模式又叫做发布-订阅(Publish/Subscribe)模式、模型-视图(Model/View)模式、源-监听器(Source/Listener)模式或从属者(Dependents)模式。

观察者模式的核心在于Subject和Observer接口,Subject(主题目标)包含一个给定的状态,观察者“订阅”这个主题,将主题的当前状态通知观察者,每次给定状态改变时所有观察者都会得到通知。

发生改变的对象称为观察目标,而被通知的对象称为观察者,一个观察目标可以对应多个观察者,而且这些观察者之间没有相互联系,可以根据需要增加和删除观察者,使得系统更易于扩展。

在PHP的标准库(SPL)里甚至提供了三个接口SplSubjectSplObserverSplObjectStorage来让开发者更容易地实现观察者模式。

模式结构说明

观察者模式

  • Subject 目标抽象类
  • ConcreteSubject 具体目标
  • Observer 观察者抽象类
  • ConcreteObserver 具体观察者

何时使用观察者模式

  • 一个对象的改变将导致其他一个或多个对象也发生改变,而不知道具体有多少对象将发生改变,可以降低对象之间的耦合度。
  • 一个对象必须通知其他对象,而并不知道这些对象是谁。
  • 基于事件触发机制来解耦复杂逻辑时,从整个逻辑的不同关键点抽象出不同的事件,主流程只需要关心最核心的逻辑并能正确地触发事件(Subject),其余相关功能实现由观察者或者叫订阅者来完成。

应用举例

<?php

class Subject implements SplSubject
{
    private $observers = [];
    private $stateNow;

    /**
     * Attach an SplObserver
     * @param SplObserver $observer
     */
    public function attach(SplObserver $observer)
    {
        // TODO: Implement attach() method.
        $this->observers[] = $observer;
    }

    /**
     * Detach an observer
     * @param SplObserver $observer
     */
    public function detach(SplObserver $observer)
    {
        // TODO: Implement detach() method.
        $key = array_search($observer, $this->observers);
        if ($key !== false) {
            unset($this->observers[$key]);
        }
    }

    /**
     * Notify an observer
     */
    public function notify()
    {
        // TODO: Implement notify() method.
        foreach ($this->observers as $observer) {
            $observer->update($this);
        }
    }

    public function setState($state)
    {
        $this->stateNow = $state;
        $this->notify();
    }

    public function getState()
    {
        return $this->stateNow;
    }
}


class EmailObserver implements SplObserver
{
    /**
     * Receive update from subject
     * @param SplSubject $subject
     */
    public function update(SplSubject $subject)
    {
        // TODO: Implement update() method.
        echo $subject->getState() . ',发送邮件' . PHP_EOL;
    }
}

class SmsObserver implements SplObserver
{
    /**
     * Receive update from subject
     * @param SplSubject $subject
     */
    public function update(SplSubject $subject)
    {
        // TODO: Implement update() method.
        echo $subject->getState() . ',发送短信' . PHP_EOL;
    }
}

// run
$emailObserver = new EmailObserver();
$smsObserver = new SmsObserver();

$subject = new Subject();
$subject->attach($emailObserver);
$subject->attach($smsObserver);
$subject->setState('用户修改密码');

 

参考:https://github.com/kevinyan815/Learning_Laravel_Kernel/blob/master/articles/Observer.md

上一篇:观察者模式实践-实现winform 窗体之间传值


下一篇:一文看懂观察者模式及案例分析