• 8 种现代 JavaScript 响应式模式


    响应性本质上是关于系统如何对数据变化作出反应,有不同类型的响应性。然而,在这篇文章中,我们关注的是响应性,即响应数据变化而采取行动。

    作为一名前端开发者,Pavel Pogosov 每天都要面对这个问题。因为浏览器本身是一个完全异步的环境。现代 Web 界面必须快速响应用户的操作,这包括更新 UI、发送网络请求、管理导航和执行各种其他任务。

    尽管人们常常将响应性与框架联系在一起,Pavel Pogosov 认为通过纯 JavaScript 实现响应性可以学到很多。所以,我们将自己编写一些模式代码,并研究一些基于响应性的原生浏览器 API。

    目录
    • PubSub(发布-订阅模式)

    • 自定义事件作为浏览器版本的 PubSub

    • 自定义事件目标

    • 观察者模式

    • 使用 Proxy 的响应式属性

    • 单个对象属性和响应性

    • 使用 MutationObserver 的响应式 HTML 属性

    • 使用 IntersectionObserver 的响应式滚动

    1. PubSub(发布-订阅模式)
    1. class PubSub {
    2.   constructor() {
    3.     this.subscribers = {};
    4.   }
    5.   subscribe(event, callback) {
    6.     if (!this.subscribers[event]) {
    7.       this.subscribers[event] = [];
    8.     }
    9.     this.subscribers[event].push(callback);
    10.   }
    11.   // 向特定事件的所有订阅者发布消息
    12.   publish(event, data) {
    13.     if (this.subscribers[event]) {
    14.       this.subscribers[event].forEach((callback) => {
    15.         callback(data);
    16.       });
    17.     }
    18.   }
    19. }
    20. const pubsub = new PubSub();
    21. pubsub.subscribe('news', (message) => {
    22.   console.log(`订阅者1收到了新闻:${message}`);
    23. });
    24. pubsub.subscribe('news', (message) => {
    25.   console.log(`订阅者2收到了新闻:${message}`);
    26. });
    27. // 向 'news' 事件发布消息
    28. pubsub.publish('news''最新头条新闻:...');
    29. // 控制台日志输出:
    30. // 订阅者1收到了新闻:最新头条新闻:...
    31. // 订阅者2收到了新闻:最新头条新闻:...

    一个常见的使用示例是 Redux。这款流行的状态管理库基于这种模式(或更具体地说,是 Flux 架构)。在 Redux 的上下文中,工作机制相当简单:

    发布者:store 充当发布者。当一个 action 被派发时,store 会通知所有订阅的组件状态的变化。 订阅者:应用程序中的 UI 组件是订阅者。它们订阅 Redux store 并在状态变化时接收更新。

    自定义事件作为浏览器版本的 PubSub

    浏览器通过 CustomEvent 类和 dispatchEvent 方法提供了一个用于触发和订阅自定义事件的 API。后者不仅能让我们触发事件,还能附加任何想要的数据。

    1. const customEvent = new CustomEvent('customEvent', {
    2.   detail: '自定义事件数据'// 将所需数据附加到事件
    3. });
    4. const element = document.getElementById('.element-to-trigger-events');
    5. element.addEventListener('customEvent', (event) => {
    6.   console.log(`订阅者1收到了自定义事件:${event.detail}`);
    7. });
    8. element.addEventListener('customEvent', (event) => {
    9.   console.log(`订阅者2收到了自定义事件:${event.detail}`);
    10. });
    11. // 触发自定义事件
    12. element.dispatchEvent(customEvent);
    13. // 控制台日志输出:
    14. // 订阅者1收到了自定义事件:自定义事件数据
    15. // 订阅者2收到了自定义事件:自定义事件数据
    自定义事件目标

    如果你不想在全局 window 对象上分派事件,可以创建你自己的事件目标。

    通过扩展原生 EventTarget 类,你可以向其新实例分派事件。这确保你的事件仅在新类本身上触发,避免了全局传播。此外,你可以直接将处理程序附加到这个特定实例上。

    1. class CustomEventTarget extends EventTarget {
    2.   constructor() {
    3.     super();
    4.   }
    5.   // 触发自定义事件的自定义方法
    6.   triggerCustomEvent(eventName, eventData) {
    7.     const event = new CustomEvent(eventName, { detail: eventData });
    8.     this.dispatchEvent(event);
    9.   }
    10. }
    11. const customTarget = new CustomEventTarget();
    12. // 向自定义事件目标添加事件监听器
    13. customTarget.addEventListener('customEvent', (event) => {
    14.   console.log(`自定义事件收到了数据:${event.detail}`);
    15. });
    16. // 触发自定义事件
    17. customTarget.triggerCustomEvent('customEvent''你好,自定义事件!');
    18. // 控制台日志输出:
    19. // 自定义事件收到了数据:你好,自定义事件!
    观察者模式

    观察者模式与 PubSub 非常相似。你订阅 Subject,然后它通知其订阅者(观察者)关于变化,使他们能够做出相应的反应。这种模式在构建解耦和灵活的架构中发挥了重要作用。

    1. class Subject {
    2.   constructor() {
    3.     this.observers = [];
    4.   }
    5.   addObserver(observer) {
    6.     this.observers.push(observer);
    7.   }
    8.   // 从列表中移除观察者
    9.   removeObserver(observer) {
    10.     const index = this.observers.indexOf(observer);
    11.     if (index !== -1) {
    12.       this.observers.splice(index, 1);
    13.     }
    14.   }
    15.   // 通知所有观察者关于变化
    16.   notify() {
    17.     this.observers.forEach((observer) => {
    18.       observer.update();
    19.     });
    20.   }
    21. }
    22. class Observer {
    23.   constructor(name) {
    24.     this.name = name;
    25.   }
    26.   // 通知时调用的更新方法
    27.   update() {
    28.     console.log(`${this.name} 收到了更新。`);
    29.   }
    30. }
    31. const subject = new Subject();
    32. const observer1 = new Observer('观察者1');
    33. const observer2 = new Observer('观察者2');
    34. // 将观察者添加到主体
    35. subject.addObserver(observer1);
    36. subject.addObserver(observer2);
    37. // 通知观察者关于变化
    38. subject.notify();
    39. // 控制台日志输出:
    40. // 观察者1 收到了更新。
    41. // 观察者2 收到了更新。
    使用 Proxy 的响应式属性

    如果你想对对象的变化做出反应,Proxy 是一个好方法。它让我们在设置或获取对象字段的值时实现响应性。

    1. const person = {
    2.   name: 'Pavel',
    3.   age: 22,
    4. };
    5. const reactivePerson = new Proxy(person, {
    6.   // 拦截设置操作
    7.   set(target, key, value) {
    8.     console.log(`将 ${key} 设置为 ${value}`);
    9.     target[key] = value;
    10.     // 表示设置值是否成功
    11.     return true;
    12.   },
    13.   // 拦截获取操作
    14.   get(target, key) {
    15.     console.log(`获取 ${key}`);
    16.     return target[key];
    17.   },
    18. });
    19. reactivePerson.name = 'Sergei'// 将 name 设置为 Sergei
    20. console.log(reactivePerson.name); // 获取 name: Sergei
    21. reactivePerson.age = 23// 将 age 设置为 23
    22. console.log(reactivePerson.age); // 获取 age: 23
    单个对象属性和响应性

    如果你不需要跟踪对象中的所有字段,可以使用 Object.defineProperty 或一组 Object.defineProperties 来选择特定的一个或几个。

    1. const person = {
    2.   _originalName: 'Pavel'// 私有属性
    3. }
    4. Object.defineProperty(person, 'name', {
    5.   get() {
    6.     console.log('获取属性 name')
    7.     return this._originalName
    8.   },
    9.   set(value) {
    10.     console.log(`将属性 name 设置为值 ${value}`)
    11.     this._originalName = value
    12.   },
    13. })
    14. console.log(person.name) // '获取属性 name' 和 'Pavel'
    15. person.name = 'Sergei' // 将属性 name 设置为值 Sergei
    使用 MutationObserver 的响应式 HTML 属性

    在 DOM 中实现响应性的一种方法是使用 MutationObserver。其 API 允许我们观察目标元素及其子元素的属性变化和文本内容变化。

    1. function handleMutations(mutationsList, observer) {
    2.   mutationsList.forEach((mutation) => {
    3.     // 观察到的元素的一个属性发生了变化
    4.     if (mutation.type === 'attributes') {
    5.       console.log(`属性 '${mutation.attributeName}' 更改为 '${mutation.target.getAttribute(mutation.attributeName)}'`);
    6.     }
    7.   });
    8. }
    9. const observer = new MutationObserver(handleMutations);
    10. const targetElement = document.querySelector('.element-to-observe');
    11. // 开始观察目标元素
    12. observer.observe(targetElement, { attributes: true });
    使用 IntersectionObserver 的响应式滚动

    IntersectionObserver API 允许对目标元素与另一个元素或视口区域的交集做出反应。

    1. function handleIntersection(entries, observer) {
    2.   entries.forEach((entry) => {
    3.     // 目标元素在视口中
    4.     if (entry.isIntersecting) {
    5.       entry.target.classList.add('visible');
    6.     } else {
    7.       entry.target.classList.remove('visible');
    8.     }
    9.   });
    10. }
    11. const observer = new IntersectionObserver(handleIntersection);
    12. const targetElement = document.querySelector('.element-to-observe');
    13. // 开始观察目标元素
    14. observer.observe(targetElement);

    感谢阅读!

    最后:

    vue2与vue3技巧合集

    VueUse源码解读

  • 相关阅读:
    node.js-包
    Linux的OpenLava配置
    java并发编程中的四个关键字:ThreadLocal、Volatile、Synchronized和Atomic
    超级计算机技术学习与研究
    物联网终端算法
    springboot实现简单的注册登录功能
    训练千亿参数大模型,离不开四种GPU并行策略
    docker入门加实战—docker常见命令
    C++笔记:类和对象(一)->封装
    淘宝/天猫API:item_search_jupage-天天特价
  • 原文地址:https://blog.csdn.net/qq449245884/article/details/139338407