Vue响应式原理学习总结3:渲染watcher

2024-01-20 13:18

本文主要是介绍Vue响应式原理学习总结3:渲染watcher,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

终于到了渲染watcher,看完这篇文章的内容后,大家就可以实现一个响应式系统了,并且能够在页面上有所体现。

源码地址:gitee

系列文章:

1. 基本原理

2. 数组的处理

4. 最终章

Vue项目总结系列文章:

  1. 基础架构
  2. 登录与权限控制

持续更新中。。。

什么是渲染Watcher

vue中有多种watcher,我们之前实现的watcher类似于Vue.$watch,当依赖变化时执行回调函数。而渲染watcher不需要回调函数,渲染watcher接收一个渲染函数而不是依赖的表达式,当依赖发生变化时,自动执行渲染函数

new Watcher(app, renderFn)
复制代码

那么如何做到依赖变化时重新执行渲染函数呢,我们要先对Watcher的构造函数做一些改造

constructor(data, expOrFn, cb) {this.data = data// 修改if (typeof expOrFn === 'function') {this.getter = expOrFn} else {this.getter = parsePath(expOrFn)}this.cb = cbthis.value = this.get()
}// parsePath的改造,返回一个函数
function parsePath(path) {const segments = path.split('.')return function (obj) {for (let key of segments) {if (!obj) returnobj = obj[key]}return obj}
}

这样,this.getter就是一个取值函数了,get修改

get() {pushTarget(this)const data = this.dataconst value = this.getter.call(data, data) // 修改popTarget()return value
}

要想依赖变化时重新执行渲染函数,就要在派发更新阶段做一个更新,因此,update方法也要进行修改:

update() {// 重新执行get方法const value = this.get()// 渲染watcher的value是undefined,因为渲染函数没有返回值// 因此value和this.value都是undefined,不会进入if// 如果依赖是对象,要触发更新if (value !== this.value || isObject(value)) {const oldValue = this.valuethis.value = valuethis.cb.call(this.vm, value, oldValue)}
}function isObject(target) {return typeof target === 'object' && target !== null
}

大家可能会有疑问了,为什么不能直接用this.getter.call(this.data)来重新执行渲染函数呢,这就涉及到下文要提到的重新收集依赖了。但是在此之前,要先解决一个问题:依赖的重复收集

重复的依赖

看这样一个例子

<div>{{ name }} -- {{ name }}
</div>

如果我们渲染这个模板,那么渲染watcher就会依赖两次name。因为解析该模板时,会读取两次name的值,就会触发两次getter,此时Dep.target都是当前watcher,在depend方法中,

depend() {if (Dep.target) {dep.addSub(Dep.target)}
}

依赖会被收集两次,name变化时就会触发两次重新渲染。因此vue采用了以下方式

首先为每个dep添加一个id

let uid = 0constructor() {this.subs = []this.id = uid++ // 增加
}
复制代码
watcher`修改的地方比较多,首先为增加四个属性`deps, depIds, newDeps, newDepIds
this.deps = []             // 存放上次求值时存储自己的dep
this.depIds = new Set()    // 存放上次求值时存储自己的dep的id
this.newDeps = []          // 存放本次求值时存储自己的dep
this.newDepIds = new Set() // 存放本次求值时存储自己的dep的id

​ 每次取值完毕后,会交换depnewDep,并将newDep清空,下文会讲到

我们的思路是,当需要收集watcher时,由watcher来决定自己是否需要被dep收集。在上面的例子中,假设对name取值时,watcherdep1收集,第二次对name取值时,watcher发现自己已经被dep1收集过了,就不会重新收集一遍,代码如下

// dep.depend
depend() {if (Dep.target) {Dep.target.addDep(this) // 让watcher来决定自己是否被dep收集}
}// watcher.addDep
addDep(dep) {const id = dep.id// 如果本次求值过程中,自己没有被dep收集过则进入ifif (!this.newDepIds.has(id)) {// watcher中记录收集自己的dpthis.newDepIds.add(id)this.newDeps.push(dep)if (!this.depIds.has(id)) {dep.addSub(this)}}
}

现在解释一下最后一个if,考虑重新渲染的情况:watcher依赖namename发生了变化,导致watcherget方法执行,会重新对name取值,进入addDep方法时,newDepIds是空的,因此会进入if,来到最后一个if,因为第一次取值时,dep已经收集过watcher了,所以不应该再添加一遍,这个if就是这个作用。

《Vue技术内幕》总结的很好:

  1. newDepsnewDepIds用来再一次取值过程中避免重复依赖,比如:{{ name }} -- {{ name }}
  2. depsdepIds用来再重新渲染的取值过程中避免重复依赖

再执行get方法最后会清空newDeps,newDepIds

cleanUpDeps() {// 交换depIds和newDepIdslet tmp = this.depIdsthis.depIds = this.newDepIdsthis.newDepIds = tmp// 清空newDepIdsthis.newDepIds.clear()// 交换deps和newDepstmp = this.depsthis.deps = this.newDepsthis.newDeps = tmp// 清空newDepsthis.newDeps.length = 0}

依赖的重新收集

我所理解的依赖重新收集包括两部分内容:收集新的依赖和删除无效依赖。其实收集新依赖再上面的代码中已经有所体现了,虽然前面的代码中对重复依赖做了很多判断,但是能够收集到依赖的基本前提是Dep.target存在,从Watcher的代码中可以看出,只有在get方法执行过程中,Dep.target是存在的,因此,我们在update方法中使用了get方法来重新触发渲染函数,而不是getter.call()。并且重新收集依赖是必要的,比如使用了v-if的情况,因此,现在的响应式系统比之前的固定依赖版本又有了很大进步。

至于删除无效依赖部分,可以在cleanUpDeps中添加如下代码

cleanUpDeps() {// 增加let i = this.deps.lengthwhile (i--) {const dep = this.deps[i]if (!this.newDepIds.has(dep.id)) {dep.removeSub(this)}}let tmp = this.depIds// ...
}

在求值结束(也就是依赖收集结束)后,如果本次求值过程中,发现有些dep在上次求值时收集了自己,但是这次求值时没有收集自己,说明该数据已经不需要自己了,将自己从dep中删除即可

// Dep.js
removeSub(sub) {remove(this.subs, sub)
}function remove(arr, item) {if (!arr.length) returnconst index = arr.indexOf(item)if (index > -1) {return arr.splice(index, 1)}
}

这样,我们的响应式系统就比较完整了

总结

其实所谓的渲染watcher和其他的watcher区别不大,只是依赖变化时自动执行渲染函数而已,上文中提到的重复依赖的处理,依赖重新收集是通用的。

下一篇文章将会做一个简单的模板编译器,让我们的响应式系统与页面渲染相结合,并且会实现v-model的双向绑定,请大家关注。

如果各位看官感觉文章还可以的话,就请点个赞吧!!!

这篇关于Vue响应式原理学习总结3:渲染watcher的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/626136

相关文章

HTML5 getUserMedia API网页录音实现指南示例小结

《HTML5getUserMediaAPI网页录音实现指南示例小结》本教程将指导你如何利用这一API,结合WebAudioAPI,实现网页录音功能,从获取音频流到处理和保存录音,整个过程将逐步... 目录1. html5 getUserMedia API简介1.1 API概念与历史1.2 功能与优势1.3

全面解析HTML5中Checkbox标签

《全面解析HTML5中Checkbox标签》Checkbox是HTML5中非常重要的表单元素之一,通过合理使用其属性和样式自定义方法,可以为用户提供丰富多样的交互体验,这篇文章给大家介绍HTML5中C... 在html5中,Checkbox(复选框)是一种常用的表单元素,允许用户在一组选项中选择多个项目。本

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

CSS3中的字体及相关属性详解

《CSS3中的字体及相关属性详解》:本文主要介绍了CSS3中的字体及相关属性,详细内容请阅读本文,希望能对你有所帮助... 字体网页字体的三个来源:用户机器上安装的字体,放心使用。保存在第三方网站上的字体,例如Typekit和Google,可以link标签链接到你的页面上。保存在你自己Web服务器上的字

SQL中JOIN操作的条件使用总结与实践

《SQL中JOIN操作的条件使用总结与实践》在SQL查询中,JOIN操作是多表关联的核心工具,本文将从原理,场景和最佳实践三个方面总结JOIN条件的使用规则,希望可以帮助开发者精准控制查询逻辑... 目录一、ON与WHERE的本质区别二、场景化条件使用规则三、最佳实践建议1.优先使用ON条件2.WHERE用

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

html 滚动条滚动过快会留下边框线的解决方案

《html滚动条滚动过快会留下边框线的解决方案》:本文主要介绍了html滚动条滚动过快会留下边框线的解决方案,解决方法很简单,详细内容请阅读本文,希望能对你有所帮助... 滚动条滚动过快时,会留下边框线但其实大部分时候是这样的,没有多出边框线的滚动条滚动过快时留下边框线的问题通常与滚动条样式和滚动行

Mysql的主从同步/复制的原理分析

《Mysql的主从同步/复制的原理分析》:本文主要介绍Mysql的主从同步/复制的原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录为什么要主从同步?mysql主从同步架构有哪些?Mysql主从复制的原理/整体流程级联复制架构为什么好?Mysql主从复制注意

Nacos注册中心和配置中心的底层原理全面解读

《Nacos注册中心和配置中心的底层原理全面解读》:本文主要介绍Nacos注册中心和配置中心的底层原理的全面解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录临时实例和永久实例为什么 Nacos 要将服务实例分为临时实例和永久实例?1.x 版本和2.x版本的区别