当前位置: 首页 > article >正文

前端开发的观察者模式

什么是观察者设计模式

观察者模式(Observer Pattern)是前端开发中常用的一种设计模式。它定义了一种一对多的依赖关系,使得当一个对象的状态发生改变时,其所有依赖对象都能收到通知并自动更新。观察者模式广泛应用于事件驱动的系统,如浏览器事件、发布-订阅系统,以及各类框架的状态管理机制。

观察者设计模式定义了一种 一对多 的依赖关系,当 一个对象的状态发生变化 时,所有依赖它的对象都会自动收到通知。这种模式可以解耦观察者和被观察者,使得它们可以独立变化。

观察者模式的关键角色:

  1. Subject(被观察者): 负责维护观察者列表,并在自身状态发生变化时通知所有观察者。
  2. Observer(观察者): 订阅目标的变化,并在收到通知时执行相关操作。

观察者模式的实现

1. 简单观察者模式

// 观察者接口
interface Observer {
  update(state: string): void;
}

// 被观察者
class Subject {
  private observers: Observer[] = [];
  private state: string = '';

  // 添加观察者
  public attach(observer: Observer): void {
    this.observers.push(observer);
  }

  // 移除观察者
  public detach(observer: Observer): void {
    this.observers = this.observers.filter(obs => obs !== observer);
  }

  // 通知所有观察者
  public notify(): void {
    this.observers.forEach(observer => observer.update(this.state));
  }

  // 改变状态并通知观察者
  public setState(state: string): void {
    this.state = state;
    this.notify();
  }
}

// 具体的观察者
class ConcreteObserver implements Observer {
  private name: string;

  constructor(name: string) {
    this.name = name;
  }

  // 收到通知时的动作
  public update(state: string): void {
    console.log(`${this.name} received update: ${state}`);
  }
}

// 测试观察者模式
const subject = new Subject();

const observer1 = new ConcreteObserver('Observer 1');
const observer2 = new ConcreteObserver('Observer 2');

subject.attach(observer1);
subject.attach(observer2);

subject.setState('State A');
subject.setState('State B');

在这个例子中,我们实现了基本的观察者模式:一个 Subject 类可以被多个 Observer 观察,每当 Subject 的状态发生变化时,它会通知所有观察者。

2. 发布-订阅模式

发布-订阅模式(Publish-Subscribe Pattern) 是观察者模式的一种变体,通常通过事件系统来实现。发布者不直接通知观察者,而是通过中介(消息中心)将事件广播出去,由订阅者选择性地接收。

// 消息中心
class EventEmitter {
  private events: { [key: string]: Function[] } = {};

  // 订阅事件
  public subscribe(event: string, listener: Function): void {
    if (!this.events[event]) {
      this.events[event] = [];
    }
    this.events[event].push(listener);
  }

  // 发布事件
  public publish(event: string, data?: any): void {
    if (this.events[event]) {
      this.events[event].forEach(listener => listener(data));
    }
  }

  // 取消订阅
  public unsubscribe(event: string, listener: Function): void {
    if (this.events[event]) {
      this.events[event] = this.events[event].filter(l => l !== listener);
    }
  }
}

// 测试发布-订阅模式
const eventEmitter = new EventEmitter();

const handler = (data: any) => console.log('Event received:', data);

// 订阅事件
eventEmitter.subscribe('eventA', handler);

// 发布事件
eventEmitter.publish('eventA', { message: 'Hello, World!' });

// 取消订阅
eventEmitter.unsubscribe('eventA', handler);

在发布-订阅模式中,发布者和订阅者是通过 EventEmitter 消息中心进行通信的,发布者无需知道订阅者的存在,这使得它们之间的耦合度进一步降低。

3. 双向绑定模式(MVVM)

MVVM(Model-View-ViewModel) 是一种常见的双向绑定模式,在前端框架如 VueAngular 中很常见。它通过数据绑定使得 ViewModel 可以双向通信,通常是通过一个 ViewModel 来桥接二者。

// ViewModel
class ViewModel {
  private state: string = '';
  private observers: Function[] = [];

  // 获取当前状态
  public getState(): string {
    return this.state;
  }

  // 设置状态并通知视图更新
  public setState(newState: string): void {
    this.state = newState;
    this.notify();
  }

  // 添加视图更新的回调
  public bind(observer: Function): void {
    this.observers.push(observer);
  }

  // 通知视图更新
  private notify(): void {
    this.observers.forEach(observer => observer(this.state));
  }
}

// 视图更新函数
const updateView = (state: string) => {
  console.log('View updated:', state);
};

// 测试双向绑定
const viewModel = new ViewModel();
viewModel.bind(updateView);

viewModel.setState('New State');
console.log('Current State:', viewModel.getState());

在这个例子中,ViewModel 充当了模型和视图之间的桥梁,确保视图在数据更新时及时反应。这是 MVVM 模式的一个简单实现。

总结

观察者设计模式及其变体在前端开发中起着至关重要的作用。通过观察者模式,可以解耦对象之间的直接依赖关系,从而提升系统的灵活性和可维护性。无论是基础的观察者模式,还是更为复杂的发布-订阅模式、双向绑定模式,都在不同的场景中发挥了不可替代的作用。

 


http://www.kler.cn/a/305813.html

相关文章:

  • 如何在 Windows 10/11 上录制带有音频的屏幕 [3 种简单方法]
  • CANN 学习——基于香橙派 KunpengPro(1)
  • 汇编实现函数调用
  • (一)使用 WebGL 绘制一个简单的点和原理解析
  • 设计模式与游戏完美开发(3)
  • 重温设计模式--13、策略模式
  • K8s 之Pod的定义及详细资源调用案例
  • NAT技术
  • 人工智能辅助汽车造型设计
  • 健身管理|基于java的健身管理系统小程序(源码+数据库+文档)
  • 数据结构与算法图论 并查集
  • 【Linux】调试和Git及进度条实现
  • 弹框调取阿里云播放器一直报错 TypeError: 没有为播放器指定容器
  • 注意!Facebook已移除细分定位排除受众的功能
  • 基于微信小程序的宠物之家的设计与实现
  • 备战软考Day02-数据结构与算法
  • 深度学习| 快速上手深度学习代码的阅读和改写
  • 6.1 溪降技术:绳结
  • 小阿轩yx-Zabbix企业级分布式监控环境部署
  • 期望极大算法(Expectation Maximization Algorithm,EM)
  • 基于SpringBoot的校园新闻网站设计与实现
  • 视觉SLAM ch5——相机与图像
  • AIGC-初体验
  • python 大模型验证码识别
  • C++11的部分新特性
  • Vue路由:Vue router