简单的响应式demo

Vue 最标志性的功能就是其低侵入性的响应式系统。组件状态都是由响应式的 JavaScript 对象组成的。当更改它们时,视图会随即自动更新。------出自vue《深入响应式系统》第一段

正文

对于熟悉或稍有了解Vue的开发者来说,响应式机制早已深入人心,其核心理念在于当数据发生变化时,能够自动更新与之相关的页面元素。

最近,在抖音上频繁地看到某前端培训机构的视频,我好奇地点击进去,并认真观看了他们的公开课。在最后的视频中有介绍到响应式,我茅塞顿开,决定记录下这些知识点并与大家分享。

在传统的HTML和JavaScript开发中,我们经常遇到一个问题:当我们通过变量和函数将数据渲染到页面上后,一旦变量值发生变化,就需要重新执行函数来更新页面。这种手动触发更新的方式既繁琐又容易出错。那么,是否有一种方法能够自动在变量值变化时调用相关的函数,从而实现页面的自动更新呢?

答案是肯定的,这就涉及到了JavaScript中的一个内置方法Object.defineProperty

该方法用于直接在一个对象上定义新属性,或者修改一个对象的现有属性。同时提供了对数据属性的更底层访问以及定义 getter 和 setter 的能力,利用这一能力,我们可以监听对象属性的变化,并在属性值发生变化时自动执行相应的函数,从而实现页面的自动更新。

语法

js 复制代码
Object.defineProperty(obj, propertyName, descriptor)
  • obj:要定义属性的对象。

  • propertyName:要定义或修改的属性的名称或 Symbol。

  • descriptor:这个属性的描述符。必须是一个对象,可以具有以下属性:

    • value :属性的值,默认为 undefined
    • writable :如果为 true,属性的值可以被重写。默认为 false
    • enumerable :如果为 true,则属性会被枚举(例如,通过 for...in 循环或 Object.keys())。默认为 false
    • configurable :如果为 true,属性可以被删除或修改。默认为 false
    • get:一个函数,作为 getter,当访问此属性时会被调用。
    • set:一个函数,作为 setter,当修改此属性时会被调用。

demo代码

在HTML文件中,我们可以定义一个输入框(<input />元素)来接收用户输入的文本。然后将第一个字符显示在#firstName中,将其余字符串显示在#lastName中。

html 复制代码
<div id="app">
  <div>
    姓名:<input id="name" />
  </div>
  <p id="firstName"></p>
  <p id="lastName"></p>
</div>
js 复制代码
// 将对象变为响应式
function reactive(obj) {
  const newObj = {
    ...obj,
  }
  for (const key in newObj) {
    let internalValue = newObj[key];
    const funcSet = new Set();
    Object.defineProperty(newObj, key, {
      get: function () {
        // 依赖收集,记录:是哪个函数在用该属性
        if (window.__func) {
          funcSet.add(window.__func);
        }
        return internalValue;
      },
      set: function (val) {
        internalValue = val;
        // 派发更新,运行:执行用该属性的函数
        funcSet.forEach(func => {
          func();
        })
      },
    });
  }
  return newObj;
}

// 准备方法
function autorun(fn) {
  window.__func = fn;
  fn();
  window.__func = null;
}

const info = reactive({
  name: '',
})

// 显示姓氏
function showFirstName() {
  document.querySelector('#firstName').textContent = '姓:' + (info.name[0] ?? '');
}

// 显示名字
function showLastName() {
  document.querySelector('#lastName').textContent = '名:' + info.name.slice(1);
}

autorun(showFirstName);
autorun(showLastName);

document.querySelector('#name').addEventListener('input', e => {
  info.name = e.target.value
})

在深入探讨响应式机制时,我们可以看到,addEventListener 并没有直接用于重新执行渲染方法(如 showFirstNameshowLastName)。相反,当对象属性被访问(get)时,与之相关的函数会被收集起来。随后,当这些属性发生更新(set)时,先前收集的函数将依次被执行,从而实现数据的自动响应和页面的动态更新。

虽然此示例代码在功能性和完整性上还有待加强,但作为展示响应式原理的入门案例,它已经足够直观和易懂。

因此,我们可以对响应式机制做出如下简洁而明确的定义:响应式是一种编程范式,它使得数据变化时能够自动触发并执行与之相关的函数,从而保持应用程序的实时性和动态性。

相关推荐
嚣张农民1 小时前
推荐3个实用的760°全景框架
前端·vue.js·程序员
落魄小二1 小时前
el-table 表格索引不展示问题
javascript·vue.js·elementui
neter.asia2 小时前
vue中如何关闭eslint检测?
前端·javascript·vue.js
十一吖i2 小时前
前端将后端返回的文件下载到本地
vue.js·elementplus
光影少年2 小时前
vue2与vue3的全局通信插件,如何实现自定义的插件
前端·javascript·vue.js
熊的猫3 小时前
JS 中的类型 & 类型判断 & 类型转换
前端·javascript·vue.js·chrome·react.js·前端框架·node.js
mosen8684 小时前
Uniapp去除顶部导航栏-小程序、H5、APP适用
vue.js·微信小程序·小程序·uni-app·uniapp
别拿曾经看以后~5 小时前
【el-form】记一例好用的el-input输入框回车调接口和el-button按钮防重点击
javascript·vue.js·elementui
Gavin_9155 小时前
【JavaScript】模块化开发
前端·javascript·vue.js
Devil枫11 小时前
Vue 3 单元测试与E2E测试
前端·vue.js·单元测试