Vue学习笔记(十三) 响应式原理

news/2024/7/10 0:07:58 标签: Vue, 响应式

1、外在表现

响应式系统是 Vue 的核心特性之一,它的实质其实就是 当数据模型发生变化时,视图将会自动更新

这样可以避免开发人员直接操作 DOM,大大降低开发难度,这也是 Vue 提倡 数据驱动 的原因

我们看一个例子

<!DOCTYPE html>
<html>
<head>
    <title>Demo</title>
    <script src="https://cdn.jsdelivr.net/npm/vue"></script>
</head>
<body>
    <div id="app">
        <div>Counter: {{ counter }}</div><br/>
        <button v-on:click="increase">Increase</button>
        <button v-on:click="decrease">Decrease</button>
    </div>
    <script>
        new Vue({
            el: '#app',
            data: {
                counter: 0
            },
            methods: {
                increase: function() { this.counter += 1 },
                decrease: function() { this.counter -= 1 }
            }
        })
    </script>
</body>
</html>

当我们点击按钮调用方法时,会使得数据 counter 增加或减少,而对应的视图 {{ counter }} 也会相应变化

整个过程,我们只需关注数据层的处理,而无需考虑视图层的渲染,这就是响应式系统带给我们的好处

2、内在原理

响应式系统的核心就是当数据模型发生变化时,视图将会自动更新,那么 Vue 它是怎么做到的呢?

(1)数据代理

每个组件实例都有一个 data 选项,当我们将一个对象传入 data 选项时,Vue 将会遍历这个对象的所有属性

并且使用 Object.defineProperty 方法将这些属性全部转换为 getter 和 setter 方法

// 模拟实现数据代理(简单版)
function proxy(data) {
    if (!data) return
    if (typeof data !== 'object') return
    Object.keys(data).forEach(function(key) { // 将对象中的所有属性转换为存取器
		defineReactive(data, key, data[key])
    })
    function defineReactive(object, key, value) { // 这里使用闭包是为了解决栈溢出问题
        proxy(value) // 递归
        Object.defineProperty(object, key, {
            enumerable: true,
            configurable: true,
            get: function() {
                console.log('访问属性', key)
                return value
            },
            set: function(new_value) {
                proxy(new_value) // 递归
                console.log('设置属性', key)
                value = new_value
            }
        })
    }
}
// 测试使用数据代理
class Component {
    constructor(options) {
        this.data = options.data
        proxy(this.data)
    }
}
let component = new Component({
    data: {
        counter: 0,
        message: 'hello',
        mapping: { 'a': 1, 'b': 3, 'c': 5 }
    }
})
console.log(component.data.mapping.a) // 访问属性 mapping -> 访问属性 a -> 1
component.data.mapping.a = 2 // 访问属性 mapping -> 设置属性 a
console.log(component.data.mapping.a) // 访问属性 mapping -> 访问属性 a -> 2

(2)依赖收集

每个组件实例都有一个 watcher 实例,它会在组件渲染过程中把接触过的 (getter 被触发的) 数据都记录为依赖

当依赖的 setter 被触发时,就会通知 watcher,使得 watcher 关联的组件重新渲染,这里放一张官方文档的图:

在这里插入图片描述

(3)组件渲染

如果侦听到数据变化,Vue 就会开启一个队列,缓冲在同一事件循环中发生的所有数据变更

在这个过程中,如果一个 watcher 它被多次触发,那么只会推入队列一次

然后在下一个事件循环中,Vue 刷新队列并执行实际的 DOM 操作

3、注意事项

(1)无法动态添加根级响应式属性

Vue 在初始化实例时,会对 data 对象中的属性执行 getter/setter 转化

所以,只有在 data 对象上的属性才能让 Vue 将它转换为响应式属性

var vm = new Vue({
    data: {
        a: 1, // 响应式属性
        object: {}
    }
}) 
vm.b = 2 // 非响应式属性

对于已经创建的实例,Vue 不允许动态添加根级别的响应式属性

但是可以使用 Vue.set(object, propertyName, value) 方法向嵌套对象添加响应式属性

Vue.set(vm.object, 'b', 2)

或者使用 vm.$set 实例方法,它也是全局 Vue.set 方法的别名

this.$set(this.object, 'b', 2)

(2)无法检测数组变化

由于 Object.defineProperty 方法的限制,在修改数组中的元素时无法触发 setter

Vue 也就无法检测到数组的变化,因此对于数组操作是非响应式

var vm = new Vue({
    data: {
        items: ['a', 'b', 'c']
    }
})
vm.items[0] = 'x' // 非响应式

这时,我们还是可以使用 Vue.set(object, propertyName, value) 方法解决这个问题

Vue.set(vm.items, 0, 'x')

另外,Vue 也提供变异的数组方法处理这种情况,变异的数组方法可以使得数组操作变成响应式

变异的数组方法包括:push()pop()shift()unshift()splice()sort()reverse()

vm.items.splice(0, 1, 'x')

http://www.niftyadmin.cn/n/1849682.html

相关文章

大数据建设意义_大数据技术平台建设方案(ppt)

应急指挥一张图可视化平台方案(ppt)大数据平台架构建设方案(图文)大数据平台技术架构解决方案(ppt)大数据平台总体架构方案(ppt)大数据平台框架选型分析方案(图文)大数据可视化分析平台应用方案(图文)大数据应用分发平台架构方案(ppt)大数据存储技术解决方案(ppt)大数据平台数据…

DataSet,DataTable,DataReader,DataAdapter区别(搞笑版)

ASP.net有这些知识,DataSet,DataTable,DataReader,DataAdapter. DataSet是用来做连接sql的一种方法,意思是把数据库的副本存在应用程序里,应用程序开始运行时,把数据库相关数据保存到DataSet. DataTable表示内存中数据的一个表.常和DefaultView使用获取可能包括筛选视图或游标位…

三角测量计算三维坐标的代码_适用于不同的人群的三维建模重建软件

Agisoft Metashape Professional for Mac是一款功能实用的三维建模重建软件&#xff0c;该软件适用于不同的人群&#xff0c;无论你是初学者&#xff0c;还是专业人士&#xff0c;这可款软件都可以满足您的各方面需要。同时该软件在配置方面相对自由&#xff0c;用户可以根据自…

[转]如何在Windows Azure虚拟机上部署OpenLogic CentOS镜像

作者信息&#xff1a;本篇文章是由OpenLogic 创始人&#xff0c;Engineering的DirectorEric Weidner所著&#xff0c;文中讲述了公司如何为 CentOS客户提供支持和服务&#xff0c;其中详细介绍了如何获取运行在Windows Azure Virtual Machines上的OpenLogic CentOS 镜像。 Open…

esp32 camera_ESP32操作L298N以及示例程序说明

本教程介绍了如何使用ESP32和L298N电机驱动器来控制DC电机的方向和速度。首先&#xff0c;我们将快速了解L298N电机驱动器的工作原理。然后&#xff0c;我们将向您展示一个示例&#xff0c;说明如何使用带有Arduino IDE的ESP32和L298N电机驱动器来控制DC电机的速度和方向。注意…

八款免费的Linux备份工具

http://os.51cto.com/art/201011/235727.htm转载于:https://blog.51cto.com/tonychiu/1137186

React学习笔记(八) 虚拟DOM与Diff算法

1、虚拟 DOM &#xff08;1&#xff09;什么是虚拟 DOM 先回顾一下什么是 DOM&#xff1f;DOM 是一个 用于表示 HTML 文档结构 的树&#xff0c;实际上它是一个 JavaScript 对象 树上的每一个节点代表一个 HTML 元素&#xff0c;每个 HTML 元素拥有大量的属性、方法和事件&a…

图书馆的uml概念类图是什么_产品经理必学UML:类图

本篇文章主要介绍了UML静态视图中的类图&#xff0c;包括类图的概念、用途及相关元素&#xff0c;供大家一起参考和学习。UML(Unified Modeling Language)又称统一建模语言或标准建模语言&#xff0c;可以看做用于系统设计阶段给开发做参考的一种方式&#xff0c;其中很多图需要…