观察者模式是“一对多”依赖关系,被观察者状态改变时自动通知所有观察者更新;PHP通过Observer和Subject接口解耦事件触发与处理,如User类变更邮箱时通知日志、推送、风控等观察者。

观察者模式的核心逻辑
观察者模式本质是“一对多”的依赖关系:当一个对象(被观察者)状态改变,所有依赖它的对象(观察者)都会自动收到通知并更新。PHP中不内置该模式,但用接口+类组合就能轻松实现,关键是解耦事件触发和处理逻辑。
定义观察者与被观察者接口
先约定契约,让代码更清晰、可扩展:
Observer接口声明每个观察者必须实现的update()方法:
interface Observer {
public function update($data);
}
Subject接口定义被观察者应提供的基础能力:
立即学习“PHP免费学习笔记(深入)”;
interface Subject {
public function attach(Observer $observer);
public function detach(Observer $observer);
public function notify();
}
实现一个可监听的用户类
比如User类注册邮箱变更事件,其他模块(如日志、推送、风控)只需作为观察者响应,无需修改User本身:
class User implements Subject {
private $observers = [];
private $email;
public function attach(Observer $observer) {
$this->observers[] = $observer;
}
public function detach(Observer $observer) {
$key = array_search($observer, $this->observers, true);
if ($key !== false) unset($this->observers[$key]);
}
public function notify() {
foreach ($this->observers as $observer) {
$observer->update(['email' => $this->email]);
}
}
public function setEmail($email) {
$this->email = $email;
$this->notify(); // 状态变更后主动广播
}
}
添加多个具体观察者
不同业务关心同一事件,各自实现update()即可:
- EmailLogger:记录操作日志
- PushService:给用户发站内信
- RiskMonitor:检查邮箱是否为高风险域名
示例:
class EmailLogger implements Observer {
public function update($data) {
error_log("Email changed to: " . $data['email']);
}
}
$user = new User();
$user->attach(new EmailLogger());
$user->attach(new PushService());
$user->setEmail("new@example.com"); // 自动触发全部观察者
事件监听机制不是魔法,而是设计选择
PHP没有像Node.js那样的原生事件循环,但观察者模式提供了轻量、可控的事件驱动风格。它适合中小型系统中松耦合的通知场景,比如订单创建后发短信、用户登录后刷新缓存、表单提交后校验权限等。注意避免观察者执行耗时操作阻塞主流程,必要时可异步投递或加队列。
基本上就这些。不复杂但容易忽略——关键在接口抽象和职责分离。











