Vue底层监测数据变化的原理

在这里插入图片描述

🌱前言 🌱

最近在学习前端,学习Vue框架,学习路线是视频+官方文档,视频看的是b站尚硅谷张天禹老师讲的,我去,讲的是太TM好了!!!不管是前端还是后端,在我看过的所有关于编程的课里面,我觉得这是我看过的最好的教程了!真的就是把深入浅出体现的淋漓尽致!很多东西他不单单是教你怎么样,他还会给你将原理讲的透透彻彻!一个例子不够,他就来两个,两个不够他就来三个…真的是把你讲的心服口服,我觉得这个真的不仅仅要自己技术过硬,然后还要把它恰到好处的讲出来,难度真的很大!张老师太赞啦!!! 👍 👍 👍
这不也是刚刚看了关于Vue数据监测的那部分内容,觉得真的太赞了!所以写篇博文记录一下,也加深一下对Vue底层的理解。

🌈关于作者 🌈

作者:白日梦想猿 🦍
作者简介:一只会做白日梦的猿 🦍!
个人博客:0318-SPACE

我们由表及里,从事物的表面现象出发,一步一步向里探索!一起来享受这个探索的过程吧!!! 🐛 🐛 🐛

👀现象 👀

使用Vue的小伙伴都知道,当我们将某一个数据修改后,页面上使用这个数据的内容也会马上变化,从而能够给用户一个及时的响应,这也是为什么说Vue是响应式(reactive)的!这是我们能够直接看到的现象,但是为什么能够产生这种现象?更直白一点就是:为什么它能够监测到数据的变化呢?

对于底层不是很懂的小伙伴可能会说:因为使用了v-bind(或者v-model)将页面上的数据和Vue中data的数据做了一个绑定,所以当Vue中data的数据发生了改变,页面上的数据也会发生改变!

看到上面回答,有的小伙伴可能会说:你这不是相当于没说嘛,就好像我问你Vue为什么可以干这件事,你回答说:因为它可以干这件事,所以它可以干这件事嘛(废话文学你是懂的 🤨)… 😑 😑 😑

🐛探索 🐛

与其只看表面现象,不如我们一步一步来自己摸索!现在我们假设有这么一个情景:

在js中定义了一个变量name,当这个name的值发生改变时就在控制台输出:name的值被改变了!

思路1

最直接最简单最容易想到的方法应该就是使用一个定时器来循环往复的监听:使用定时器定时监测name的值,一旦name的值发生变化就打印日志。
代码实现如下:

 let name = '张三'let tmp = name //中间变量,用来和name进行比较,判断name的值是否发生变化//每隔100毫秒做一次检测setInterval(() => {if (name !== tmp) {tmp = nameconsole.log('name的值被改变了!')}}, 100)

运行结果如下:
在这里插入图片描述
通过上面的例子我们可以看到,当我们对定义的数据进行修改时,后台能够很快监测到我们的动作,并及时的响应,做到了响应式的效果!这会不会是Vue的底层实现呢?在做出结论之前,再来看一下另一种思路!

思路2

使用get和set来实现,当需要读取时自动调用get;当数据修改时自动调用set。

在开始第二种思路前,我们得先了解一个内置方法,它就是Object.defineProperty,一个强大的方法!

Object.defineProperty方法介绍

从方法名可以看到,它要实现的功能很直观,就是定义属性!没错,它的功能就是给指定的一个对象定义一个属性,已经属性的相关操作。

语法
<script>
Object.defineProperty(obj,xxx,{...})
/*
* obj:要接受的一个对象
* xxx:属性名
* {...}:属性的相关配置
* 配置项中有两个重要的方法,分别是:get()和set(v)
*/
</script>
例子

给一个对象添加一个age属性,当读取和修改age属性时分别打印相应的提示信息。我这里定义了一个对象person,它自身有name和address属性,然后通过Object.defineProperty方法为它定义了一个age属性。值得注意的是:age属性的读取和修改都是通过配置方法中的get和set来实现的!并且:当age被读取时,get会被自动调用;当age发生变化时,set会被自动调用!咦?这个set方法不相当于一个现成的“监测”方法吗?没错!它就是一个现成的监测方法!相关代码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Object.defineProperty方法</title><script src="js/vue.js"></script>
</head>
<body>
<div id="root"></div>
<script>let person = {name: 'Kim','address': 'China',// 'age':18}//给对象添加属性let num = 18Object.defineProperty(person, 'age', {// value: 18, //值enumerable: true, //控制属性是否可以枚举,默认值为false,即默认不会被枚举// writable: true, //控制属性是否可被修改,默认值为false,即默认不可被修改// configurable: false,//控制属性是否可被删除,默认值为false,即默认不可被删除//当有人读取当前定义的属性时,get就会被调用,读取到的就是get的返回值get() {console.log('age被读取了!')return num},//当有人修改当前定义的属性时,set就会被调用set(v) {console.log('age被修改了!新的age:', v)num = v}})
</script>
</body>
</html>

运行结果如下:
在这里插入图片描述

可以看到,使用Object.defineProperty方法能够更加完美的实现我们想要的需求

实现
	let person = {name: 'jerry',age: 18}Object.defineProperty(person, 'name', {get() {console.log('get被调用了!')return person.name},set(val) {console.log('set被调用了,name的值被修改为:' + val)person.name = val}})

似乎已经找到了完美答案,但是从执行结果来看:不管是对name的值进行读取还是修改,都会发生栈溢出!居然还直接报错了!
在这里插入图片描述
来看一下它内部的调用顺序,也许就可以找到问题出在哪里了!
在这里插入图片描述
可以看到,不管是对name的值进行读取还是修改,都会重复调用自己,进而导致栈溢出!如果此时我们再联想一下vue里面的_data也许就会有答案了:对于vue中的vm对象,它既有data属性也有_data属性,并且data里面的属性,_data里面也都有,而且不管是对data还是_data里面的值进行修改,另外一个里面的值也会同步发生变化!

模拟实现Vue的数据监测

自己实现的不足

1.数据操作麻烦

2.只能实现一层

优化

总结

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/139314.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

全自动打包机检测不到货物怎么办?

全自动打包机也称无人化打包机或无人化捆扎机&#xff0c;是指在工作的过程中不需要人工干预&#xff0c;机器可以自动检测、自动输送、自动打包的设备。但最近有一些客户反映打包机不能自己检测到货物了。这是为什么呢&#xff1f; 1、电子眼感应太弱。电子眼的感应程度是可以…

使用目标之间的先验关系提升目标检测器性能

今天跟大家分享阿姆斯特丹大学等提出的用于提升目标检测和实例分割性能的新方法RP-FEM&#xff0c;该方法将目标之间位置的先验关系融入到feature中。 论文标题&#xff1a;Relational Prior Knowledge Graphs for Detection and Instance Segmentation机构&#xff1a;阿姆斯特…

RabbitMQ 安装和使用Demo

同步是阻塞 MQ&#xff1a;消息队列&#xff0c;基础数据结构中"先进先出"的数据结构。用来解决应用解耦&#xff0c;异步消息&#xff0c;流量消峰等问题。 RabbitMQ、RocketMQ、Kafka RocketMQ 是阿里的 应用层开发使用RabbitMQ 大数据开发Kafka MQ是不是微服务都…

「爱嗨游」发布会剧透|应用出海十大难题,全景方案一次解决

融云“融云出海 & 全球化通信方案发布会”&#xff08;&#x1f448; 点击报名&#xff09;开幕在即&#xff0c;一波宝藏产品、惊喜方案、创意玩法即将喷薄而出。关注【融云全球互联网通信云】了解更多 ❤ Global IM UIKit&#xff0c;社交 全场景解决方案&#xff0c;北极…

oracle 错题汇总

在Oracle数据库中&#xff0c;如果你在GROUP BY子句中指定了列&#xff0c;那么SELECT子句中除了聚合函数外的每个列都必须在GROUP BY子句中列出。 1.查询出每个部门的编号、名称、位置、部门人数、平均工资。 SELECT d.deptno AS "部门编号", d.dname AS &…

Vue3+ElementPlus el-date-picker时间选择器,设置最多选择60天区间内,必须选择大于今天

Vue3ElementPlus el-date-picker时间选择设置&#xff0c; 必须选择大于今天&#xff1b;最多选择60天区间内&#xff1b; 使用disabled-date&#xff1a; 具体实现代码&#xff1a; <el-date-pickerv-model"state.queryParams.dateTime":editable"false&q…

基础课5——语音合成技术

TTS是语音合成技术的简称&#xff0c;也称为文语转换或语音到文本。它是指将文本转换为语音信号&#xff0c;并通过语音合成器生成可听的语音。TTS技术可以用于多种应用&#xff0c;例如智能语音助手、语音邮件、语音新闻、有声读物等。 TTS技术通常包括以下步骤&#xff1a; …

Bootstrap的媒体对象组件(图文展示组件),挺有用的一个组件。

Bootstrap的.media类是用于创建媒体对象的&#xff0c;媒体对象通常用于展示图像&#xff08;图片&#xff09;和文本内容的组合&#xff0c;这种布局在展示新闻文章、博客帖子等方面非常常见。.media类使得创建这样的媒体对象非常简单&#xff0c;通常包含一个图像和相关的文本…

竞赛选题 深度学习YOLO抽烟行为检测 - python opencv

文章目录 1 前言1 课题背景2 实现效果3 Yolov5算法3.1 简介3.2 相关技术 4 数据集处理及实验5 部分核心代码6 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于深度学习YOLO抽烟行为检测 该项目较为新颖&#xff0c;适合作为竞赛课…

液压泵马达试验台比例阀控制器

添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09; 液压泵马达试验台主要应用于液压元件性能测试、故障诊断、产品研发等领域。 液压泵马达试验台主要功能包括排量验证试验、容积效率试验、总效率试验、变量特性试验、超载试验、冲击性能试验、启动效率试验…

Flink的算子列表状态的使用

背景 算子的列表状态是平时比较常见的一种状态&#xff0c;本文通过官方的例子来看一下怎么使用算子列表状态 算子列表状态 算子列表状态支持应用的并行度扩缩容&#xff0c;如下所示: 使用方法参见官方示例&#xff0c;我加了几个注解&#xff1a; public class Bufferin…

MySQL的ACID和并发事务带来的问题简单总结

拓跋阿秀 ACID 原子性&#xff1a;一个事务&#xff08;transaction&#xff09;中的所有操作&#xff0c;要么全部完成&#xff0c;要么全部不完成&#xff0c;不会结束在中间某个环节。事务在执行过程中发生错误&#xff0c;会被恢复&#xff08;Rollback&#xff09;到事务…