观察者模式通过分离数据源与处理逻辑,实现模块解耦。Subject维护Observer列表,状态变化时通知所有观察者执行相应逻辑。以用户注册为例,User类作为Subject在注册后触发事件,EmailService和LogService作为Observer接收通知并发送邮件、记录日志。通过接口定义attach、detach和notify方法,支持动态增删观察者,避免硬编码依赖。结合事件调度器可按事件类型绑定监听器,提升灵活性。实际应用中建议使用SPL接口、容器管理观察者、异步处理耗时任务,并及时清理无效引用,防止内存泄漏。该模式符合开闭原则,适用于一变多的场景,有效分离发布与订阅逻辑。

在PHP开发中,当模块之间存在强依赖,数据变化需要通知多个组件时,使用观察者模式能有效解耦。它让数据源(被观察者)与处理逻辑(观察者)分离,提升代码可维护性和扩展性。
观察者模式的基本结构
观察者模式包含两个核心角色:
- Subject(主题/被观察者):维护观察者列表,状态变化时主动通知所有观察者。
- Observer(观察者):实现统一接口,接收通知并执行对应逻辑。
通过接口约定,Subject无需知道具体观察者类型,实现松耦合。
实现一个简单的事件通知系统
以用户注册后发送邮件和日志记录为例,展示如何用观察者模式解耦业务逻辑。
立即学习“PHP免费学习笔记(深入)”;
interface Observer {
public function update($event, $data);
}
interface Subject {
public function attach(Observer $observer);
public function detach(Observer $observer);
public function notify($event, $data);
}
class User implements Subject {
private $observers = [];
private $name;
public function __construct($name) {
$this->name = $name;
}
public function attach(Observer $observer) {
$this->observers[] = $observer;
}
public function detach(Observer $observer) {
$this->observers = array_filter(
$this->observers,
fn($o) => $o !== $observer
);
}
public function register() {
// 模拟注册逻辑
echo "用户 {$this->name} 注册成功\n";
// 触发事件
$this->notify('user.registered', ['name' => $this->name]);
}
public function notify($event, $data) {
foreach ($this->observers as $observer) {
$observer->update($event, $data);
}
}
}
class EmailService implements Observer {
public function update($event, $data) {
if ($event === 'user.registered') {
echo "发送欢迎邮件给 {$data['name']}\n";
}
}
}
class LogService implements Observer {
public function update($event, $data) {
echo "日志记录:用户 {$data['name']} 已注册\n";
}
}
使用示例:
$user = new User("张三");
$user->attach(new EmailService());
$user->attach(new LogService());
$user->register();
结合事件机制提升灵活性
可以进一步封装事件调度器,支持按事件类型绑定不同观察者,类似现代框架的事件系统。
- 定义事件名称,如 user.login、order.created。
- 允许动态注册监听器,便于插件化扩展。
- 避免在主流程中硬编码调用通知逻辑。
这样新增功能只需添加新观察者,不影响原有代码,符合开闭原则。
实际应用中的优化建议
在真实项目中,可结合以下方式增强实用性:
- 使用SPL的 SplSubject 和 SplObserver 接口,遵循PHP标准。
- 将观察者注册过程配置化或通过容器管理,降低耦合。
- 异步处理耗时操作(如发短信),可通过队列实现。
- 注意内存泄漏,及时 detach 不再需要的观察者。
基本上就这些。观察者模式特别适合处理“一变多”的场景,让数据变化的发布与消费完全分离,是解耦事件驱动逻辑的有效手段。











