PHP设计模式——观察者模式
- 2019 年 11 月 8 日
- 筆記
定义对象之间的一对多依赖关系,以便在对象更改状态时,将自动通知和更新其所有依赖关系。
玛丽刚刚成为新妈妈!像所有新妈妈一样,她对新男婴内森(Nathan)感到非常兴奋。生活给玛丽带来了极大的收获,但同时也充满了挑战。作为一个新妈妈,她正在经历照顾新生婴儿的喜悦和挑战。婴儿是一个轻便的睡眠者,很容易受到大人发出的小声音的干扰。为了确保婴儿能够充分休息和快乐,玛丽和她的丈夫决定将他们的客房变成托儿所。玛丽这做的一点点牺牲对婴儿来说是完美的,因为她必须在晚上多次检查婴儿以确保一切都很好。
从编码的角度来看,这像以下这样:
$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
类:
interface Observable { public function registerObserver(Observer $o); public function removeObserver(Observer $o); public function notifyObservers(); }
Observer
类:
interface Observer { public function update(); }
然后我们需要修改Son
类以实现Observable
接口:
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
接口:
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
的观察员:
$mary = new Mother(); $nathan = new Son(); $nathan->registerObserver($mary);
在观察者模式(Observer Pattern
)的帮助下,玛丽改善了自己的日常生活,现在更享受妈妈的生活。在某些夜晚,她甚至根本不需要醒来。随着Nathan的成长,Mary当然将不再需要使用婴儿音频监视器。那是她可以解除自己作为儿子的旁观者的时候:
$nathan->removeObserver($mary);