写几行代码,了解响应式原理

作者:袁首京

原创文章,转载时请保留此声明,并给出原文连接。

作为当下的开发人员,无论是不是前端,可能都会频繁的与 React、Vue、Svelte、Solidjs 等等打交道。也许你已经很清楚它们背后的运作原理,那不必往下看了。如果还不是很清楚,那咱们可以一起写几行代码,来瞅一眼这些响应式框架背后的思路。

响应式框架最根本的功能其实只有一条:当数据发生变化时,让界面随之发生变化

如何达成这一点呢?粗略的想一下就会觉得,首先要在数据和与之对应的 HTML 元素之间建立绑定关系。可以以某种方式给特定的 HTML 元素打个标记,然后当与此元素相关的值发生变更时,我们就能通过这个标记找到此元素,然后动态的改变它展示出来的值。

比如如下 HTML 模板片断:

<p>{{ current_time }}</p>

我们可以定义一个模板编译函数:

function compile(tpl) {
  const re = /(\{\{\s+)(\w+)(\s+\}\})/m;
  const mg = tpl.match(re);
  return tpl.replace(">{{", ' vid="' + mg[2] + '">{{').replace(mg[0], "");
}

执行该函数,就会给相关元素打上 vid 标记:

> compile('<p>{{current_time}}</p>')
<p vid="current_time"></p>

这样如果需要,我们就可以很方便的找到页面上需要响应的元素:

const vel = document.querySelector("[vid=current_time]");

接下来是数据部分。如何监测数据的变化呢?一种方案是使用代理。假如我们有如下数据对象:

{
  current_time: "2023-05-03T05:14:46.176Z";
}

可以使用如下函数,为其生成一个代理,拦截其赋值操作:

function reactive(data) {
  return new Proxy(data, {
    set(target, property, value) {
      const prev = target[property];
      target[property] = value;

      if (prev !== value) {
        const vel = document.querySelector(`[vid=${property}]`);
        vel.innerHTML = value;
      }

      return true;
    },
  });
}

接下来,就可以面向数据编程了:

const data = reactive({
  current_time: "2023-05-03T05:14:46.176Z",
});

setInterval(() => {
  data.current_time = new Date().toISOString();
}, 1000);

最终效果如下:

写几行代码,了解响应式原理

以下是完整代码:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
  </head>
  <body>
    <div id="root"></div>
    <script>
      function compile(tpl) {
        const re = /(\{\{\s+)(\w+)(\s+\}\})/m;
        const mg = tpl.match(re);
        return tpl.replace(">{{", ' vid="' + mg[2] + '">{{').replace(mg[0], "");
      }

      function reactive(data) {
        return new Proxy(data, {
          set(target, property, value) {
            const prev = target[property];
            target[property] = value;

            if (prev !== value) {
              const vel = document.querySelector(`[vid=${property}]`);
              vel.innerHTML = value;
            }

            return true;
          },
        });
      }

      const app = {
        tpl: "<p>{{ current_time }}</p>",

        data: {
          current_time: "2023-05-03T05:14:46.176Z",
        },

        mount() {
          const rootEl = document.querySelector("#root");
          rootEl.innerHTML = compile(this.tpl);

          this.data = reactive(this.data);

          this.mounted();
        },

        mounted() {
          setInterval(() => {
            this.data.current_time = new Date().toISOString();
          }, 1000);
        },
      };

      document.addEventListener("DOMContentLoaded", () => {
        app.mount();
      });
    </script>
  </body>
</html>

原文链接:https://www.cnblogs.com/rockety/p/17369256.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:写几行代码,了解响应式原理 - Python技术站

(0)
上一篇 2023年4月30日
下一篇 2023年5月4日

相关文章

  • JavaScript类库D

    JavaScript类库D完整攻略 什么是JavaScript类库D JavaScript类库D是一款基于JavaScript语言的开源类库,提供了丰富的工具函数和组件,可以大大提高开发效率。它的主要特点包括以下几点: 支持多种浏览器和平台; 提供了丰富的工具函数和组件,包括DOM操作、Ajax、动画、事件绑定等; 提供了易于扩展和定制的接口。 如何使用Ja…

    JavaScript 2023年6月10日
    00
  • JavaScript的内置对象Date详解

    JavaScript的内置对象Date详解 1. Date对象概述 Date对象是JavaScript的内置对象,它封装了时间和日期相关的方法。使用Date对象,可以获取当前的日期和时间,还可以进行日期和时间的运算以及格式化输出。该对象提供的方法非常丰富,能够满足大部分与时间有关的需求。 2. 创建Date对象 Date对象可以通过以下两种方式进行创建: 2…

    JavaScript 2023年5月27日
    00
  • ES6 javascript中Class类继承用法实例详解

    ES6 javascript中Class类继承用法实例详解 1. 什么是ES6中的Class类 在ES6(ES2015)中,我们可以使用Class关键字来定义一个类,这是一种更加面向对象的编程方法,使得代码更加易读、易维护。使用Class关键字定义类后,我们可以通过关键字new来创建该类的实例。 下面是一个简单的示例: class Person{ const…

    JavaScript 2023年6月11日
    00
  • js数组循环遍历数组内所有元素的方法

    当我们需要操作一个数组内的所有元素时,循环遍历就是最基本的方法之一。 使用for循环 for 循环是最常用的循环语句之一,可以很方便地遍历数组中的所有元素。 const arr = [0, 1, 2, 3, 4, 5]; for (let i = 0; i < arr.length; i++) { console.log(arr[i]); } 上述代码…

    JavaScript 2023年5月27日
    00
  • JavaScript实现设置默认日期范围为最近40天的方法分析

    要设置默认日期范围为最近40天,可以通过以下步骤实现: 1.使用JavaScript获取当前时间 首先,我们需要获取当前时间作为默认结束时间。我们可以使用 JavaScript 的 Date() 函数获取当前时间: let now = new Date(); let endDate = now.toISOString().substr(0, 10); 这里,…

    JavaScript 2023年6月10日
    00
  • JS实现判断数组是否包含某个元素示例

    下面我将详细讲解如何实现判断数组是否包含某个元素。 1. 使用includes()方法判断数组是否包含某个元素 includes() 方法用来判断一个数组是否包含一个指定的值,如果包含则返回 true,否则返回 false。下面是使用 includes() 方法判断数组是否包含某个元素的示例代码: const arr = [1, 2, 3, 4, 5]; c…

    JavaScript 2023年5月27日
    00
  • JS 实现请求调度器

    让我们来详细讲解一下“JS 实现请求调度器”的完整攻略。 什么是请求调度器 请求调度器是一个用于处理并发请求的工具,它主要用于解决并发请求的限制问题。通常,浏览器发送的并发请求数量是有限制的,如果我们需要发送多个请求,会存在发生阻塞的情况。因此,使用请求调度器可以让我们管理并发请求的数量,保证同时只有一定数量的请求被发送,从而更好地管理请求。 实现请求调度器…

    JavaScript 2023年6月11日
    00
  • js实现旋转大风车

    下面是“JS实现旋转大风车”的完整攻略。 步骤一:HTML结构 首先,我们需要在HTML中编写一个容器div和两个风车翅膀的图片。具体代码如下: <div id="windmill"> <img class="blade" src="blade.png"> <img c…

    JavaScript 2023年6月11日
    00
合作推广
合作推广
分享本页
返回顶部