观察者模式是一种行为型设计模式,定义对象间一对多依赖关系,当被观察者状态改变时,所有观察者自动收到通知并更新;php中可通过spl接口或自定义subject/observer接口实现,核心是attach、detach、notify及update方法。

什么是观察者模式
观察者模式是一种行为型设计模式,定义了对象间一对多的依赖关系。当一个对象(被观察者)状态发生改变时,所有依赖于它的对象(观察者)都会自动收到通知并更新。在 PHP 面试中,常被用来考察对设计模式的理解、面向接口编程能力以及解耦思维。
PHP 中如何实现观察者模式
PHP 原生提供了 SplSubject 和 SplObserver 接口(自 PHP 5.1 起),可直接基于标准库实现。但更常见的是手写接口+类的方式,体现理解深度。
- 定义 Observer 接口,含
update()方法,接收被观察者和可选数据 - 定义 Subject 接口,含
attach()、detach()、notify() - 实现具体被观察者类(如
NewsPublisher),维护观察者数组,触发通知时遍历调用每个观察者的update() - 实现具体观察者类(如
EmailSubscriber、SmsSubscriber),专注自身响应逻辑
面试常考细节与易错点
面试官可能追问以下问题,需提前准备:
-
是否支持传参? ——
notify()最好支持传递上下文(如事件名、新状态),避免观察者主动拉取数据 -
如何防止重复添加观察者? —— 使用
array_unique()或检查对象引用(in_array($observer, $this->observers, true)) -
是否支持移除指定观察者? —— 提供
detach()方法,支持传入对象或键名 -
异常处理怎么做? ——
notify()中用 try-catch 包裹单个观察者调用,避免一个失败中断全部通知 -
PHP 内置 Spl 实现有何限制? ——
SplSubject::notify()不接受参数,扩展性差;实际项目中更倾向自定义接口
一个简洁可运行的示例
以下代码无依赖、可直接运行,体现核心逻辑:
立即学习“PHP免费学习笔记(深入)”;
interface Observer {
public function update($subject, $data = null);
}
interface Subject {
public function attach(Observer $observer);
public function detach(Observer $observer);
public function notify($data = null);
}
class UserActivity implements Subject {
private array $observers = [];
public function attach(Observer $observer): void {
$this->observers[] = $observer;
}
public function detach(Observer $observer): void {
$key = array_search($observer, $this->observers, true);
if ($key !== false) unset($this->observers[$key]);
}
public function notify($data = null): void {
foreach ($this->observers as $observer) {
$observer->update($this, $data);
}
}
}
class EmailLogger implements Observer {
public function update($subject, $data): void {
echo "Email: 用户行为已记录 — " . json_encode($data) . "\n";
}
}
class ActivityCounter implements Observer {
private int $count = 0;
public function update($subject, $data): void {
$this->count++;
echo "Counter: 当前累计 {$this->count} 次操作\n";
}
}
// 使用示例
$activity = new UserActivity();
$activity->attach(new EmailLogger());
$activity->attach(new ActivityCounter());
$activity->notify(['action' => 'login', 'user_id' => 123]);
$activity->notify(['action' => 'logout', 'user_id' => 123]);











