PHP设计模式——观察者模式

2019-11-08 22:13:55 浏览数 (1)

定义对象之间的一对多依赖关系,以便在对象更改状态时,将自动通知和更新其所有依赖关系。

玛丽刚刚成为新妈妈!像所有新妈妈一样,她对新男婴内森(Nathan)感到非常兴奋。生活给玛丽带来了极大的收获,但同时也充满了挑战。作为一个新妈妈,她正在经历照顾新生婴儿的喜悦和挑战。婴儿是一个轻便的睡眠者,很容易受到大人发出的小声音的干扰。为了确保婴儿能够充分休息和快乐,玛丽和她的丈夫决定将他们的客房变成托儿所。玛丽这做的一点点牺牲对婴儿来说是完美的,因为她必须在晚上多次检查婴儿以确保一切都很好。

从编码的角度来看,这像以下这样:

代码语言:javascript复制
$mary   = new Mother();
$nathan = new Son();
$time   = date('H');
// 在 1点 , 3点 和 5点 起来照看儿子
if ($time=='01' || $time=='03' || $time=='05') {
    $mary->wakeUp();
    if ($nathan->isCrying()) {
        $mary->comfortBaby();
    }
}

婴儿在他的新托儿所(把客房改成托儿所)里过得很好,但她的丈夫注意到玛丽看上去有点累。“亲爱的,你为什么今天早上看起来这么累?”他问妻子。玛丽解释说,她昨晚必须多次去婴儿室,以确保婴儿状况良好。显然,这种每天晚上凌晨起床检查婴儿状况的习惯对玛丽来说已经很累了。她失眠了,早上感到疲倦。玛丽越是白天睡不着,精疲力尽,越是无法照顾婴儿,这只是时间问题。这可以带走所有新生的乐趣。她知道自己需要找到一种方法来改善这一点,因为她可能会感到压力甚至生病。幸运的是,她的好朋友罗斯(Rose)来访,并向她介绍了一种称为婴儿音频监视器的工具。音频监视器放在婴儿的房间内,当小小的Nathan哭泣或大惊小怪时,音频监视器发出声音告诉玛丽她的婴儿需要她。玛丽对这种新工具非常满意,因为她可以主动地睡个觉,而不必主动醒来检查婴儿,只有在监护仪告诉她需要的情况下,才可以醒来检查Nathan。玛丽一无所知,在这里使用了观察者模式(Observer Pattern )。让我们看看它如何应用。玛丽在婴儿音频监视器的帮助下,将自己注册为该主题的观察者(内森)。每当对象的状态发生变化(Nathan哭泣或发出任何声音)时,都会自动通知观察者(玛丽)。

让我们重新编写代码。

首先,我们需要创建两个接口。 Observable接口(是被观察者Subject的接口)和Observer接口(是观察者Observer的接口)。

Observable类:

代码语言:javascript复制
interface Observable
{
    public function  registerObserver(Observer $o);
    public function  removeObserver(Observer $o);
    public function  notifyObservers();
}

Observer类:

代码语言:javascript复制
interface Observer
{
    public function update();
}

然后我们需要修改Son类以实现Observable接口:

代码语言:javascript复制
class Son implements Observable
{
    private $_observers = array();
    private $_isCry = false;
    public function registerObserver(Observer $o)
    {
        $this->_observers[] = $o;
    }
 
    public function removeObserver(Observer $o)
    {
        foreach($this->_observers as $index=>$value) {
            if ($value==$o) {
                unset($this->_observers[$index]);
            }
        }
    }
 
    public function notifyObservers()
    {
        foreach($this->_observers as $observer) {
            $this->_observer->update();
        }
    }
 
    public function cry()
    {
        $this->_isCry = true;
        $this->notifyObservers();
    }
 
    public function isCry()
    {
        return $this->_isCry;
    }
}

Mother类需要实现Observer接口:

代码语言:javascript复制
class Mother implements Observer
{
    public function update()
    {
        $this->wakeUp();
        $this->comfortBaby();
    }
 
    public function wakeUp()
    {
        echo 'Wake up at once!';
    }
 
    public function comfortBaby()
    {
        echo 'Baby do not cry.';
    }
}

其次,让我们在运行时将Mary注册为Nathan的观察员:

代码语言:javascript复制
$mary   = new Mother();
$nathan = new Son();
$nathan->registerObserver($mary);

在观察者模式(Observer Pattern)的帮助下,玛丽改善了自己的日常生活,现在更享受妈妈的生活。在某些夜晚,她甚至根本不需要醒来。随着Nathan的成长,Mary当然将不再需要使用婴儿音频监视器。那是她可以解除自己作为儿子的旁观者的时候:

代码语言:javascript复制
$nathan->removeObserver($mary);

0 人点赞