理清Vue响应式系统中的Watcher

2023-11-11 23:50

本文主要是介绍理清Vue响应式系统中的Watcher,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

理清Vue响应式系统中的Watcher

一、什么是响应系统中的Watcher,它的作用是什么?

响应系统中的Watcher即这个系统的观察者,它是响应系统中观察者模式的载体,当响应系统中的数据发生改变的时候,它能够知道并且执行相应的函数以达到某种业务逻辑的目的。打个比方,如果你是一个商家,要寄一批货分别给不同的客户,那么watcher就是一个个快递员,发出的动作就是数据发生改变。你只需要负责寄出去这个动作就行了,如何找到、送到客户则是watcher的事情。

在这里插入图片描述

每个watcher和数据之间的关系要么是1对1,要么是多对多关系(这与watcher的类型有关),watcher和业务逻辑只有1对1关系。

二、Watcher的类型

Vue源码中是没有体现出Watcher的类型的,我在这里给Watcher添加类型是为了更好地理解Watcher这个对象。Watcher在普通的业务逻辑上可以分为以下三类:

  • 普通的Watcher:与数据1对1关系。
  • lazyWatcher:与数据1对1关系,但是它是一个惰性求值的观察者,怎么体现呢?对它进行赋值是不会改变它的值,只有当获取它的值的时候,才会更新最新版的数据(在Vue中体现为computed方法,一般求值是通过方法来求值的)。
  • renderWatcher:与数据是1对多(不考虑传参进子组件)的关系,在一个组件中,渲染函数观察者一定是最后生成的,所以执行观察者队列的时候,渲染函数观察者在一个组件中是最后执行的。

在这里多嘴一下lazy型的观察者是怎么回事吧。lazy观察者里面有一个dirty属性,也就是一个开关作用,在添加完依赖后,如果lazy观察者所引用的数据发生改变后,会将这个放到观察者队列中,然后执行的时候把dirty赋值为true(代表下次访问getter方法的时候会执行一遍lazy的求值方法(求值后会将dirty赋值为false))。这样就实现了懒求值和减少不必要的执行次数。

三、Watcher和Dep的关系

看过Vue源码的defineReactive这个方法,就会发现一个被观察的对象里面每个属性会有一个Dep依赖筐来存放所有观察它的Watcher。而defineReactive方法只有初始化每个属性的dep却并没有创建观察者(要分清初始化和创建观察者是分开这个事实)。那么这个Dep如何添加观察者呢?Vue使用了全局变量,这个变量叫做Dep.target,它是一个Watcher类型的变量,来将WatcherDep进行互相绑定。数据的绑定用图来表示的话如下:

在这里插入图片描述

我们可以明确以下区别:

  • $watch方法定义的观察者,如果不设定immediate属性,那么是不会进行调用的,而computedrender是会进行调用方法的。
  • 数据的Depsubs数组存放这个数据所绑定的观察者对象,观察者对象的deps数组中存放着与这个观察者有关的数据Dep。所以数据的DepWatcher其实是多对多关系
  • $watchcomputed观察者是在created生命钩子函数前就创建完毕并且绑定的,而render观察者是在mounted之前创建并绑定的,所以同一个组件中,render观察者的id会大于其他观察者(id是在后面执行队列里面升序排序的时候的依据)。**换句话说,同一个组件,当数据发生改变的时候,渲染函数观察者一定是最后执行的。**这个很好理解,其他观察者中难免会对数据进行修改,如果渲染函数观察者先执行了,然后其他观察者对数据进行改变的话,那么没办法将数据准确呈现在页面上,导致数据不一致性。

四、讲一下观察者执行队列机制

Vue是如何实现性能优化的呢?最显著的两个点:

  • 观察者设定执行队列,批量执行。
  • diff算法减少渲染开支。

第二个不在这里面讲解,我们看一下第一个是怎么回事?

这个队列的长度是怎么定量的呢?

  • 最大长度是100,源码摆在那里。

在这里插入图片描述

  • 以一个事件循环时间段为搜集时间。(什么是事件循环?可以看一下本博客系统的其他优秀文章)

它的流程是如下的:

  • 未执行时候:如果有更改过数据,那么就将对应的观察者直接推进队列中(执行的时候会进行根据id升序排序后执行)
  • 在执行中的时候,如果有新的观察者进来了(观察者中更改数据,然后这个数据又绑定观察者),按照序号升序来进行插入操作(最后的结果也是排序好的了)。

附录: 这边给出阅读源码时候,比较详细的DepWatcher的关系图在这里插入图片描述

这篇关于理清Vue响应式系统中的Watcher的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

前端高级CSS用法示例详解

《前端高级CSS用法示例详解》在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交互和动态效果的关键技术之一,随着前端技术的不断发展,CSS的用法也日益丰富和高级,本文将深... 前端高级css用法在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交

Python将博客内容html导出为Markdown格式

《Python将博客内容html导出为Markdown格式》Python将博客内容html导出为Markdown格式,通过博客url地址抓取文章,分析并提取出文章标题和内容,将内容构建成html,再转... 目录一、为什么要搞?二、准备如何搞?三、说搞咱就搞!抓取文章提取内容构建html转存markdown

在React中引入Tailwind CSS的完整指南

《在React中引入TailwindCSS的完整指南》在现代前端开发中,使用UI库可以显著提高开发效率,TailwindCSS是一个功能类优先的CSS框架,本文将详细介绍如何在Reac... 目录前言一、Tailwind css 简介二、创建 React 项目使用 Create React App 创建项目

vue使用docxtemplater导出word

《vue使用docxtemplater导出word》docxtemplater是一种邮件合并工具,以编程方式使用并处理条件、循环,并且可以扩展以插入任何内容,下面我们来看看如何使用docxtempl... 目录docxtemplatervue使用docxtemplater导出word安装常用语法 封装导出方

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

Linux系统中卸载与安装JDK的详细教程

《Linux系统中卸载与安装JDK的详细教程》本文详细介绍了如何在Linux系统中通过Xshell和Xftp工具连接与传输文件,然后进行JDK的安装与卸载,安装步骤包括连接Linux、传输JDK安装包... 目录1、卸载1.1 linux删除自带的JDK1.2 Linux上卸载自己安装的JDK2、安装2.1

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

Vue中组件之间传值的六种方式(完整版)

《Vue中组件之间传值的六种方式(完整版)》组件是vue.js最强大的功能之一,而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用,针对不同的使用场景,如何选择行之有效的通信方式... 目录前言方法一、props/$emit1.父组件向子组件传值2.子组件向父组件传值(通过事件形式)方

css中的 vertical-align与line-height作用详解

《css中的vertical-align与line-height作用详解》:本文主要介绍了CSS中的`vertical-align`和`line-height`属性,包括它们的作用、适用元素、属性值、常见使用场景、常见问题及解决方案,详细内容请阅读本文,希望能对你有所帮助... 目录vertical-ali

如何解决Spring MVC中响应乱码问题

《如何解决SpringMVC中响应乱码问题》:本文主要介绍如何解决SpringMVC中响应乱码问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring MVC最新响应中乱码解决方式以前的解决办法这是比较通用的一种方法总结Spring MVC最新响应中乱码解