Vue深入响应性原理

2023-11-16

参考资料

响应性

深入响应性原理

Vue最独特的特性之一,是其非侵入性对的响应性系统。数据模型是被代理的JavaScript对象。而当你修改它们时,视图会进行更新。折让状态管理非常简单直观。在本章节,将研究一下Vue响应性系统底层的细节。

什么是响应性

这个术语在程序设计中经常被提及,响应性是一种允许我们以声明式的方式去适应变化的编程范例。在JavaScript中,通常不是如此工作:

let val1=2
let val2=3
let sum=val1+val2

console.log(sum)// 5

val1=3

console.log(sum)// 仍然是 5

上面代码中,sum并没有对val1的更改做出响应,在JavaScript中,如何实现响应性,作为一个高阶的概述,需要做到以下几点:

  1. 当一个值被读取时进行追踪,例如val1+val2会同时读取两个值
  2. 当某个值改变时进行检测,例如,当我们赋值val1=3
  3. 重新运行代码来读取原始值,例如,再次运行sum=val1+val2来更新sum的值

Vue如何知道哪些代码在执行

为了能够在数值变化时,随时运行总和,首先要做的是将其包裹在一个函数中。

const updateSum=()=>{
sum=val1+val2
}

但我们如何告知Vue这个函数呢,Vue通过一个副作用(effect)来跟踪当前正在运行的函数。副作用是一个函数的包裹器,在函数被调用之前就启动跟踪。Vue知道哪个副作用在何时运行,并能在需要时再次执行它。

为了更好理解这点,让我们尝试脱离Vue实现类似的东西,看看它如何工作。

我们需要一个能包裹总和的东西,如下

createEffect(()=>{
sum=val1+val2
})

我们需要createEffect来跟踪和执行。实现如下:

//维持一个执行副作用的栈
const runningEffects=[]

const createEffect=fn=>{
	//将传来的fn包裹在一个副作用函数中
	const effect=()=>{
		runningEffects.push(effect)
		fn()
		runningEffects.pop()
	}
	//立即自动执行副作用
	effect()
}

当我们的副作用被调用时,在调用fn之前,它会把自己推到runningEffects数组中。这个数组可以用来检查当前正在运行的副作用。

副作用是许多关键功能的起点。例如,组件的渲染和计算属性都在内部使用副作用。任何时候,只要有事物对数据变化做出回应,就可以肯定它已经被包裹在一个副作用中了。

虽然Vue的公开API不包括任何直接创建副作用的方法,但它确实暴露了一个叫做watchEffect的函数,与createEffect类似。

但知道什么代码在执行只是解决了一部分。Vue如何知道副作用使用了什么值,以及如何知道它们何时发生变化?

Vue如何跟踪变化

我们不能像前面的例子中那样跟踪局部变量的重新分配,在JavaScript中没有这样的机制。我们可以跟踪的是对象property的变化。

当我们从一个组件的data函数中返回一个普通的JavaScript对象时,Vue会将该对象包裹在一个带有getset处理程序的Proxy中。Proxy是在ES6中引入的,它是Vue3避免了Vue早期版本中存在的一些响应性问题。

Proxy是一个对象,它包装了另一个对象,并允许你拦截该对象的任何交互

const dinner={
meal:'tacos'
}

const handler={
get(target,property){
console.log('intercepted!')
return target[property]
}
}

const proxy=new Proxy(dinner,handler)
console.log(proxy.meal)

// intercepted!
// tacos

这里我们截获了读取模板对象property的行为。像这样的处理函数也称为一个捕捉器(trap)。有许多可用的不同类型的捕捉器,每个都处理不同类型的交互。

使用Proxy的一个难点是this绑定。我们希望任何方法都绑定到这个Proxy,而不是目标对象,这样我们也可以拦截它们。ES6引入了另一个Reflect特性,允许我们以小代价解决这个问题:

const dinner={meal:'tacos'}

const handler={
	get(target,property,receiver)
	{
		return Reflect.get(...argument)
	}
}

const proxy = new Proxy(dinner, handler)
console.log(proxy.meal)

// tacos

使用Proxy实现响应性的第一步就是跟踪一个property何时被读取。我们在一个名为track的处理器函数中执行此操作,该函数可以传入targetproperty两个参数。

const dinner = {
  meal: 'tacos'
}

const handler = {
  get(target, property, receiver) {
    track(target, property)
    return Reflect.get(...arguments)
  }
}

const proxy = new Proxy(dinner, handler)
console.log(proxy.meal)

// tacos

这里没有展示track的实现。它将检查当前运行的是哪个副作用,并将其与targetproperty记录在一起。这就是Vue如何知道这个property是该副作用的依赖项。

最后,我们需要在property值更改时重新运行这个副作用。为此,我们需要在代理上使用一个set处理函数:


const dinner = {
  meal: 'tacos'
}

const handler = {
  get(target, property, receiver) {
    track(target, property)
    return Reflect.get(...arguments)
  },
  set(target, property, value, receiver) {
    trigger(target, property)
    return Reflect.set(...arguments)
  }
}

const proxy = new Proxy(dinner, handler)
console.log(proxy.meal)

// tacos

至此,就可以满足实现响应性的几点要求:

  1. 当一个值被读取时进行追踪
    proxy的get处理函数中track函数记录了该property和当前副作用。
  2. 当某个值改变时进行检测
    在proxy上调用set处理函数。
  3. 重新运行代码来读取原始值
    trigger函数查找哪些副作用依赖于该property并执行它们。

如果我们要用一个组件重写原来求和例子,可以这样做:

const vm=createApp({
	data(){
		return{
			val1:2,
			val2:3
		},
	computed:{
		sum(){
				return this.val1+this.val2
			}
		}
	}
}).mount('#app')

console.log(vm.sum)//5

vm.val1=3
cosole.log(vm.sum)//6

data返回的对象被包裹在响应式代理中,并存储为this.$data。propertythis.val1this.val2分别是this.$data.val1this.$data.val2的别名,因此它们通过相同的代理。

Vue将把sum的函数包裹在一个副作用中。当我们试图访问this.sum时,将运行该副作用来计算数值。$data周围的响应式代理将追踪propertyval1val2在该副作用运行时的读取。

从Vue3开始,我们的响应性现在可以在一个独立包中使用。将$data包裹在一个代理中的函数被称为reactive。我们可以自己直接调用这个函数,允许我们在不需要使用组件的情况下将一个对象包裹在一个响应式代理中。

const proxy=reactive({
	val1:2,
	val2:3
})
被代理的对象

Vue在内部跟踪所有已被转为响应式的对象,所以它总是为同一个对象返回相同的代理。

当从一个响应式代理访问一个嵌套对象时,该对象在被返回之前也会被转换为一个代理

const handler = {
  get(target, property, receiver) {
    track(target, property)
    const value = Reflect.get(...arguments)
    if (isObject(value)) {
      // 将嵌套对象包裹在自己的响应式代理中
      return reactive(value)
    } else {
      return value
    }
  }
  // ...
}
Proxy vs 原始标识

如何让渲染响应变化

一个组件的模板被编译成一个render函数。渲染函数创建VNodes,描述组件该如何被渲染。它被包裹在一个副作用中,允许Vue在运行时跟踪被“触及”的property。

一个render函数在概念上与一个computedproperty非常相似。Vue并不确切地追踪依赖关系是如何被使用的,它只知道在函数运行的某个时间点上使用了这些依赖关系。如果这些property中的任何一个随后发生了变化,它将触发副作用再次运行,重新运行render函数以生成新的VNodes。

响应性基础

声明响应式状态

要为JavaScript对象创建响应式状态,可以使用reactive方法:

import {reactive} from 'vue'

const state=reactive({
	count:0
})

reactive相当于Vue2.x中的Vue.observable()API。

Vue中响应式状态的基本用例是我们可以在渲染期间使用它。因为依赖跟踪的关系,当响应式状态改变时视图会自动更新。

这就是Vue响应性系统的本质。当从组件中的data()返回一个对象时,它在内部交由reactive()使其成为响应式对象。模板会被编译成能够使用这些响应式property的渲染函数

创建独立的响应式值作为refs

想象一下,我们有一个独立的原始值(例如,一个字符串),我们想让他变成响应式的。当然,我们可以创建一个拥有相同字符串property的对象,并将其传递给reactive。Vue为我们提供了一个可以做相同事情的方法—ref

import {ref} from 'vue'
const count=ref(0)

ref会返回一个可变的响应式对象,该对象作为一个响应式的引用维护着内部的值,这就是ref名称的来源。该对象只包含一个名为value的property

Ref解包

当ref作为渲染上下文(从setup()中返回的对象)上的property返回并可以在模板中被访问时,它将自动浅层解析包内部值。只有访问嵌套的ref时需要在模板中添加.value

<template>
  <div>
    <span>{{ count }}</span>
    <button @click="count ++">Increment count</button>
    <button @click="nested.count.value ++">Nested Increment count</button>
  </div>
</template>

<script>
  import { ref } from 'vue'
  export default {
    setup() {
      const count = ref(0)
      return {
        count,

        nested: {
          count
        }
      }
    }
  }
</script>

TIP 如果你不需要访问实际的对象实例,可将其用reactive包裹:

nested:reactive({
	count
})
访问响应式对象

ref作为响应式对象的property被访问或更改时,为使其行为类似于普通property,它会自动解包内部值:

const count=ref(0)
const state=reactive({
	count
})

console.log(state.count)//0

state.count=1
console.log(count.value)//1

如果将新的ref赋值给现有ref的property,将会替换旧的ref:

const otherCount = ref(2)

state.count = otherCount
console.log(state.count) // 2
console.log(count.value) // 1

Ref解包仅发生在被响应式Object嵌套的时候。当从 Array 或原生集合类型如 Map访问 ref 时,不会进行解包:

const books = reactive([ref('Vue 3 Guide')])
// 这里需要 .value
console.log(books[0].value)

const map = reactive(new Map([['count', ref(0)]]))
// 这里需要 .value
console.log(map.get('count').value)

响应式状态解构

当我们想使用大型响应式对象的一些property时,可能很想使用ES6解构来获取我们想要的property:

import {reactive} from 'vue'

const book=reactive({
  author: 'Vue Team',
  year: '2020',
  title: 'Vue 3 Guide',
  description: 'You are reading this book right now ;)',
  price: 'free'
})

let{author,title}=book

遗憾的是,使用解构的两个property的响应性都会丢失。对于这种情况,我们需要将我们的响应式对象转换为一组ref。这些ref将保留对源对象的响应式关联:

import { reactive, toRefs } from 'vue'

const book = reactive({
  author: 'Vue Team',
  year: '2020',
  title: 'Vue 3 Guide',
  description: 'You are reading this book right now ;)',
  price: 'free'
})

let { author, title } = toRefs(book)

title.value = 'Vue 3 Detailed Guide' // 我们需要使用 .value 作为标题,现在是 ref
console.log(book.title) // 'Vue 3 Detailed Guide'

使用readonly防止更改响应式对象

有时我们想跟踪响应式对象 (refreactive) 的变化,但我们也希望防止在应用程序的某个位置更改它。例如,当我们有一个被 provide 的响应式对象时,我们不想让它在注入的时候被改变。为此,我们可以基于原始对象创建一个只读的 proxy 对象:

import { reactive, readonly } from 'vue'

const original = reactive({ count: 0 })

const copy = readonly(original)

// 通过 original 修改 count,将会触发依赖 copy 的侦听器

original.count++

// 通过 copy 修改 count,将导致失败并出现警告
copy.count++ // 警告: "Set operation on key 'count' failed: target is readonly."

响应式计算和侦听

计算值

有时我们需要依赖于其他组件的状态,在Vue中,这是用组件计算属性处理的,以直接创建计算值,我们可以使用computed方法:它接受getter函数并为getter返回的值返回一个不可变的响应式ref对象。

const count=ref(1)
const plusOne=computed(()=>count.value+1)

console.log(plusOne.value)// 2
plusOne.value++// error

或者,它可以使用一个带有 get 和 set 函数的对象来创建一个可写的 ref 对象。

const count = ref(1)
const plusOne = computed({
  get: () => count.value + 1,
  set: val => {
    count.value = val - 1
  }
})

plusOne.value = 1
console.log(count.value) // 0

watchEffect

为了根据响应式状态自动应用和重新应用副作用,我们可以使用watchEffect方法。它立即执行传入的一个函数,同时响应式追踪其依赖,并在依赖变更时重新运行该函数。

const count=ref(0)
watchEffect(()=>console.log(count.value))
停止侦听

watchEffect在组件的setup()函数或生命周期钩子被调用时,侦听器会被链接到该组件的生命周期,并在组件卸载时自动停止。
在一些情况下,也可以显示调用返回值以停止侦听:

const stop = watchEffect(() => {
  /* ... */
})

// later
stop()
清除副作用

有时副作用函数会执行一些异步的副作用,这些响应需要在其失效时清除(即完成之前状态已经改变了)。所以侦听副作用传入的函数可以接收一个onInvalidate函数作入参,用来注册清理失效时的回调。当以下情况发生时,这个失效回调会被触发:

  • 副作用即将重新执行时
  • 侦听器被停止(如果在setup()或生命周期钩子函数中使用了watchEffect,则在组件卸载时)
watchEffect(onInvalidate => {
  const token = performAsyncOperation(id.value)
  onInvalidate(() => {
    // id has changed or watcher is stopped.
    // invalidate previously pending async operation
    token.cancel()
  })
})

watch

侦听单个数据源
侦听多个数据源
侦听响应式对象
与watchEffect共享的行为

渲染机制和优化

虚拟DOM

Vue2中的更改检测警告

该页面仅适用于Vue2.x及更低版本

由于JavaScript的限制,有些Vue无法检测的更改类型。但是,有一些方法可以规避他们以维持响应性。

对于对象
Vue无法检测到property的添加或删除。由于Vue在实例初始化期间执行getter/setter转换过程,因此必须在data对象中存在一个property,以便Vue对其进行转换并使其具有响应式。

对于已经创建的实例,Vue不允许动态添加根级别的响应式property。但是,可以使用Vue.set(object,propertyName,value)方法向嵌套对象添加响应式property:

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

还可以使用this.$set实例方法,这也是全局Vue.set方法的别名。

有时你可能需要为已有对象赋值多个新property,比如使用Object.assign()_.extent()。但是,这样添加到对象上的新property不会触发更新。在这种情况下,你应该用原对象与要混合进去的对象的property一起创建一个新的对象。

// 而不是 `Object.assign(this.someObject, { a: 1, b: 2 })`
this.someObject = Object.assign({}, this.someObject, { a: 1, b: 2 })

对于数组

Vue不能检测到以下数组的变动:

  1. 当利用索引直接设置一个数组项时,例如:vm.items[indexOfItem]=newValue
  2. 当修改数组的长度时

为了解决第一个问题,有以下两种方式:

//Vue.set
Vue.set(vm.items,indexOfItem,newValue)
//Array.prototype.splice
vm.items.splice(indexOfItem,1,newValue)

也可以使用vm.$set实例方法,该方法是全局方法Vue.set的一个别名

为了解决第二种问题,可以使用splice

vm.item.splice(newLength)

声明响应式property

由于Vue不允许动态添加根级响应式property,所以你必须在初始化实例前声明所有根级响应式property,哪怕只是一个空值:

var vm = new Vue({
  data: {
    // 声明 message 为一个空值字符串
    message: ''
  },
  template: '<div>{{ message }}</div>'
})
// 之后设置 `message`
vm.message = 'Hello!'

异步更新队列

Vue在更新DOM时是异步执行的。只要监听到数据变化,Vue将开启一个队列,并缓冲在同一事件循环中发生的所有数据变更。如果同一个侦听器被多次触发,它只会被推入队列中一次。这种在缓冲时去除重复数据对于避免不必要的计算和DOM操作是非常重要的。然后在下一个的事件循环tick中,Vue刷新队列并执行实际(已去重)的工作。

例如,当你设置vm.someData='new value',该组件不会立即重新渲染。当刷新队列时,组件会在下一个事件循环tick中更新。如果想基于更新后的DOM状态来做些什么,为了在数据变化之后等待Vue完成更新DOM,可以在数据变化之后立即使用Vue.nextTick(callback)。这样回调函数将在DOM更新完成之后被调用。例如:

<div id="example">{{ message }}</div>
var vm = new Vue({
  el: '#example',
  data: {
    message: '123'
  }
})
vm.message = 'new message'              // 更改数据
vm.$el.textContent === 'new message'    // false
Vue.nextTick(function() {
  vm.$el.textContent === 'new message'  // true
})

在组件内使用vm.$nextTick()实际方法特别方便,因为它不需要全局Vue,并且回调函数中的this将自动绑定到当前组件实例上:

Vue.component('example', {
  template: '<span>{{ message }}</span>',
  data: function() {
    return {
      message: 'not updated'
    }
  },
  methods: {
    updateMessage: function() {
      this.message = 'updated'
      console.log(this.$el.textContent)   // => 'not updated'
      this.$nextTick(function() {
        console.log(this.$el.textContent) // => 'updated'
      })
    }
  }
})

因为$nextTick()返回一个Promise对象,所以可以使用新的ES2017 async/await语法完成相同事情:


 methods: {
    updateMessage: async function () {
      this.message = 'updated'
      console.log(this.$el.textContent) // => 'not updated'
      await this.$nextTick()
      console.log(this.$el.textContent) // => 'updated'
    }
  }
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Vue深入响应性原理 的相关文章

  • VSCode:Remote-SSH配置实录

    转自 VSCode Remote SSH配置实录 六天 CSDN博客 也可以通过这样一步步输入用户名和密码链接 为什么要使用VSCode Remote SSH 服务器很多时候都是部署在Linux远程机器上的 我们通常是SSH连过去然后用vi
  • JDBC——BasicDAO

    为什么要有 BasicDAO apache dbutils Druid 简化了 JDBC 开发 但还有不足 SQL语句是固定 不能通过参数传入 通用性不好 需要进行改进 更方便执行增删改查 对于 select 操作 如果有返回值 返回类型不

随机推荐

  • Putty配色方案(转)

    平时用Putty的频率相对挺高的 每次装完系统或是怎么的都得重新配色 还得百度去找配色表 每次太麻烦了 特地转载一篇好看的配色表供以后长期使用 以下内容为转载内容 使用的是修改注册表的方法 1 打开注册表 运行 regedit 2 找到对应
  • matlab仿真实例100题_输入-输出反馈线性化(Feedback linearization)控制算法Matlab仿真实例...

    反馈线性化 Feedback linearization 可能是大部分人接触非线性控制之后学习的第一种控制方法 个人认为主要优点有两点 一 它的理解和实现都相对简单 二 它能让非线性系统的转换为线性系统 这样接下来 就可以直接套用线性系统中
  • SQLEXPRESS服务无法启动

    一 软件版本 SQL Sever2014 localdb MSSQLLocalDB SQLServer13 0 VS2015 二 问题描述 由于使用web API中的odata编程 在工程中使用的是 localdb MSSQLLocalDB
  • Lintcode 464. 整数排序 II 冒泡排序三种实现 直接插入排序 直接选择排序 java

    一 冒泡排序 实现思路 https blog csdn net morewindows article details 6657829 Lintcode https www lintcode com problem sort integer
  • Ubuntu下安装及卸载JDK

    安装 1 添加 PPA repository 到系统 sudo add apt repository ppa webupd8team java 2 更新 sudo apt get update 3 下载安装 JDK sudo apt get
  • 算法专题之二叉树

    前言 树型数据结构广泛存在于现实世界中 比如家族族谱 企业职能架构等 它是一类在计算机领域被广泛应用的非线性数据结构 二叉树是树型数据中最常用的一类 本文从前端角度以js语法构建和操作二叉树的相关知识 基础概念 观察上图 二叉树的数据结构如
  • C++——大数加法

    大数加法 即运算的数据可能很大 int long long long无法存放 存在字符串中 但是加法的运算规则还是10进制 对于两个字符串 首先判断两者的长度 我们将字符串s设置为较长的字符串 方便后面的运算 也可以将t设置为较长的 从低位
  • 一文2000字详细介绍Android APP 常见概念与 adb 命令

    01 背景 因 team 正在推进 APP 自动化测试 那么主要业务是 IoT 相关的产品 既然是 APP 自动化测试 那么 Android APP 相关的一些概念与知识就很关键了 在之前遇到相关的问题都是现查现用 对于一些常用的概念和命令
  • 【Redis】常见数据结构及命令

    Redis常见命令 Redis数据结构介绍 Redis是一个key value的数据库 key一般是String类型 但是value的类型有很多种 Redis 通用命令 通用指令是部分数据类型的 都可以使用的指令 常见的有 KEYS 查看符
  • C++实现英语词典

    需要的东西 1 一个词典的txt文件 2 创建map容器 map
  • 最小二乘支持向量机”在学习偏微分方程 (PDE) 解方面的应用(Matlab代码实现)

    欢迎来到本博客 博主优势 博客内容尽量做到思维缜密 逻辑清晰 为了方便读者 座右铭 行百里者 半于九十 本文目录如下 目录 1 概述 2 运行结果 3 参考文献 4 Matlab代码实现 1 概述 本代码说明了 最小二乘支持向量机 在学习偏
  • 多媒体(视频容量计算)-软件设计(四十二)

    真题详解 有向图 软件设计 四十一 https blog csdn net ke1ying article details 129942490 一 音频概念 声音带宽 人耳 20Hz 20KHz 说话300 3400Hz 小于20Hz次声波
  • 上海链节科技:什么是企业链改?

    如何定义链改 链改 即用区块链中的部分技术原理和机制来改造业务系统中特定的部分 以创造更高效 更有公信力的系统 甚至还包括利用区块链上的各类积分 Token通证 来重新分配业务系统进而间接的改进业务系统背后的生产关系 传统基本结构的公司 通
  • 从IDataReaderHelper中读取数据实体

    using System using System Collections Generic using System Linq using System Text using System Data using System Reflect
  • 截取oracle字符串中的数字(转载)

    截取oracle字符串中的数字 云淡风轻博客 博客园 cnblogs com 方法一 如果Oracle版本不是太低的话 使用 正则表达式函数 REGEXP SUBSTR 处理 REGEXP SUBSTR有5个参数 分别是 第一个是输入的字符
  • GMAPPING的参数设置

    二 运行gmapping 我总结了运行gmapping的两种方法 1 基于命令行 rosrun gmapping slam gmapping scan scan delta 0 1 maxUrange 4 99 xmin 5 0 ymin
  • python必背代码-Python高手必修课:如何让 Python 代码更易读,推荐收藏

    阅读本文大概需要 10 分钟 我们知道 Python 是一种动态语言 在声明一个变量时我们不需要显式地声明它的类型 例如下面的例子 a 2 print 1 a 1 a 运行结果 1 a 3 这里我们首先声明了一个变量 a 并将其赋值为了 2
  • 【深入理解C++】string

    文章目录 1 概述 2 string的定义与初始化 3 string的赋值 4 string的读写 5 string的长度 6 string的遍历 6 1 运算符 6 2 at 6 3 迭代器 6 4 范围for 7 string 与 ch
  • Excel开发(VBA)— 快速定位最后有记录的行

    有时候 在进行Excel开发时 需要知道某列的最后一行的行号 Excel也提供了相关功能 Range End属性 Dim strCell As String Dim nLastUseRow As Long Dim nLastRcdRow A
  • Vue深入响应性原理

    深入响应性原理 参考资料 响应性 深入响应性原理 什么是响应性 Vue如何知道哪些代码在执行 Vue如何跟踪变化 被代理的对象 Proxy vs 原始标识 如何让渲染响应变化 响应性基础 声明响应式状态 创建独立的响应式值作为refs Re