理解VUE响应式原理[通俗易懂]

理解VUE响应式原理[通俗易懂]1、响应式原理基础响应式原理基础是基于Object.defineProperty(obj,prop,descriptor),descriptor里面可以定义get和set方法,可以在获取属性值事触发get方法(可以收集依赖),设置属性值时触发set方法(更新依赖)。扩展:上面是vue2.0的基础原理,vue3.0的基础原理是:2、核心对象:Dep与WatcherDep:vue在data里申明的每一个属性都会生成一个Dep的实例对象,De…

大家好,又见面了,我是你们的朋友全栈君。

1、响应式原理基础

        响应式基本原理是基于Object.defineProperty(obj, prop, descriptor), descriptor里面可以定义get和set方法,可以在获取属性值事触发get方法(可以收集依赖),设置属性值时触发set方法(更新依赖)。

        扩展:上面是vue2.0的响应式基本原理,vue3.0的基本原理是Proxy,可以监听属性的get和set方法,监听属性的添加和删除等等,比Object.defineProperty能力更强,但是不兼容IE11。

2、核心对象:Dep与Watcher

        Dep: vue在data里申明的每一个属性都会生成一个Dep的实例对象,Dep.subs存储着当该属性变化时需要去更新的Watcher;

        Watcher: 有3种情况会生成Watcher的实例对象,分别为:

  1.  定义在computed里的计算属性;
  2. 在watch里写的监听函数;
  3. 组件的渲染Watcher;

3、收集依赖与更新依赖

3.1 收集依赖

        将Watcher的实例对象w分发到它所依赖的属性的Dep中,过程如下:

  1. 将Dep.target = 当前的Watcer 的实例对象w;
  2. w执行定义的函数(即在computed/watch写的函数);
  3. 执行函数的过程如果使用data里定义的属性,则会触发属性的get方法,get方法中Dep实例对象dep会将Dep.target中存储的w放入到dep.subs数组中,完成依赖收集。

    说明:Dep.target为当前Watcer的实例对象

3.2 更新依赖

        当修改我们申明的某个属性时,会触发属性的set方法,set方法会将dep.subs数组中收集的Watcher实例对象进行更新,即触发我们定义在computed和watch里面的函数。

4、源码调试

4.1 测试的页面代码

<template>
  <div>
    <div>a:<input v-model="a" /></div>
    <div>c:{
  
  { c }}</div>
    <div>b:<input v-model="b" /></div>
  </div>
</template>
<script>
export default {
  data: () => {
    return {
      a: '',
      b: ''
    }
  },
  computed: {
    c() {
      return 'source from ' + this.a;
    }
  },
  watch: {
    b() {
      console.log('b changed');
    }
  }
};
</script>

上面的代码vue初始化后会生成如下几个对象:

1、对象说明

属性a和b对应的Dep实例对象(收集a、b改变需要更新的Watcher):depA、depB;

页面渲染函数生成对应的Watcher实例对象updateWatcher;

computed属性c生成对应的Watcher实例对象:watcherC;

watch监听属性b生成对应的Watcher实例对象:watcherB;

2、Dep与Watcher的关系

a、b变化页面需要重新渲染,所以updateWatcher存在于depA和depB的subs中;

计算属性c依赖属性a的变化更更新,所以watcherC存在于depA的subs中;

b的变化会触发定义watch 里b的监听函数,所以watcherB存在于depB的subs中;

3、最终的关系结果

最终属性a收集的依赖 depA.subs = [ updateWatcher,  watcherC];

最终属性b收集的依赖 depB.subs = [ updateWatcher,  watcherB];

4.2  源码调试

        找到源码文件:node_modules\vue\dist\vue.runtime.esm.js;

        4.2.1 主要涉及如下几个函数:

        1、收集依赖的入口函数:initState(页面初始化时执行);

        初始化顺序是先data–>computed–>watch:原因是computed依赖data, watch监听data和computed里的属性,被依赖的需要先初始化。

理解VUE响应式原理[通俗易懂]

        2、初始化computed和watch时,生成Watcher实例化对象

              先执行Watcher.get函数,将Dep.target = 当前Watcher实例化对象

理解VUE响应式原理[通俗易懂]

理解VUE响应式原理[通俗易懂]

        触发收集依赖

理解VUE响应式原理[通俗易懂]

执行计算属性里面的函数,如果访问到data中的某个属性时,会触发data属性的get方法,触发依赖收集:

 理解VUE响应式原理[通俗易懂]

 当修改这个属性时会触发set方法,会触发更新dep.subs里面watcher对象

理解VUE响应式原理[通俗易懂]

最终触发Watcher的更新函数,将待更新的watcher放入队列中:

理解VUE响应式原理[通俗易懂]

4.2.2 computed属性特别说明

        computed属性在初始化initComputed之前还有一次更早的初始化,在调用createComponent函数生成vnode组件时,调用了Vue.extend –》defineComputed方法,并在组件的原型Prototype上添加computed里的计算属性,并声明为响应式属性,为何这样设计,原因不知,如有小伙伴知道,还请告知,谢谢

理解VUE响应式原理[通俗易懂]

理解VUE响应式原理[通俗易懂]

理解VUE响应式原理[通俗易懂]

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/132033.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 源代码分析:LayoutParams的wrap_content, match_parent, 而详细的价值观

    源代码分析:LayoutParams的wrap_content, match_parent, 而详细的价值观

    2022年1月16日
    92
  • Linux中的pycharm配置问题「建议收藏」

    Linux中的pycharm配置问题「建议收藏」1.pycharm的图标问题解决方法:在终端输入以下命令行:sudogedit/usr/share/applications/Pycharm.desktop进入gedit文档界面然后将里面的内容复制成:[DesktopEntry]Type=ApplicationName=PycharmGenericName=Pycharm3Comment=Pycharm3:…

    2022年8月26日
    8
  • 万能乘法速算法大全_小学生两位数乘法容易出错?只因没掌握这个“万能”速算法…

    万能乘法速算法大全_小学生两位数乘法容易出错?只因没掌握这个“万能”速算法…儿童节快乐两位数乘法,在小学阶段的数学学习当中,是经常遇到的。尤其是小学三、四年级,每当遇到这类乘法,小学生都非常容易出错,甚至一算就错。原因是,一些比较大的两位数,在用列竖式法相乘时,会出现连续进位,一旦有一个环节,进位失误了,就会导致整个乘法出现错误,结果为零分。因此,这类运算,很让老师头疼。那么,有没有实用的速算法,可以解决这类乘法问题,而且不出错呢?有!今天,就分享一种“万能”两…

    2022年6月7日
    37
  • 郑州java面试难吗_java开发人员培训机构

    郑州java面试难吗_java开发人员培训机构前言周末花了2天时间学习了额RabbitMQ,总结了最核心的知识点,带大家快速掌握RabbitMQ,整理不易希望帮忙点赞,转发,分享下,谢谢目录进入SpringBoot世界讲述Sping、SpringBoot和SpringCloud之间的关系,还重点讲述了如何利用开发工具(如IDEA)来实现开发,如何通过API文档来寻找类对象方法,告诉我们在开发过程中如何学习、发现和解决问题需要免费领取这份Alibaba架构师耗时一年整理的《SpringBoot实战,让你的开发像搭积木一样简单

    2022年9月28日
    2
  • linux 如何安装whl文件,linux安装whl文件

    linux 如何安装whl文件,linux安装whl文件Windows环境:安装whl包:pipinstallwheel->pipinstall**.whl下载whl文件MySQL_python-1.2.5-cp27-none-win32.whlMySQL_python-1.2.5-cp27-none-win_amd64.whl执行pipinstallMySQL_python-1.2.5-cp27-none-win3…

    2022年5月30日
    44
  • list对象转map[通俗易懂]

    list对象转map[通俗易懂]根据list对象中的某个属性转换成map/***将对象中的某个属性作为map的key将对象本身作为map的value构成成一个map**@paramfieldToKey必须是obj的field我们把field的getValue作为map的key*@authormountain2019-01-0717:21*/publicstatic<T,E>Map<T,E>listToM

    2022年5月13日
    78

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号