简单的响应式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)时,先前收集的函数将依次被执行,从而实现数据的自动响应和页面的动态更新。

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

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

相关推荐
lvbb661 小时前
框架修改思路
前端·javascript·vue.js
qq_456001651 小时前
43、接口请求需要时间,导致页面初始加载时会出现空白,影响用户体验
javascript·vue.js·ux
工业互联网专业1 小时前
基于springboot+vue的动漫交流与推荐平台
java·vue.js·spring boot·毕业设计·源码·课程设计·动漫交流与推荐平台
FanetheDivine1 小时前
实现"选中表格项将元素加入集合"的动画效果
javascript·vue.js
Json_1 小时前
Vue 构造器 Vue.extend
前端·vue.js·深度学习
小old弟1 小时前
老生常谈: 谈谈对vue的理解?vue3有而vue2没有的东西或者说特性?
前端·vue.js
Json_1 小时前
Vue 实例方法
前端·vue.js·深度学习
Json_2 小时前
Vue Methods Option 方法选项
前端·vue.js·深度学习
刘 怼怼3 小时前
使用 Vue 重构 RAGFlow 实现聊天功能
前端·vue.js·人工智能·重构
Json_3 小时前
Vue v-bind指令
前端·vue.js·深度学习