vue的响应式原理

  当你把一个普通的 JavaScript 对象传入 Vue 实例作为 data 选项,Vue 将遍历此对象所有的属性,并使用 Object.defineProperty 把这些属性全部转为 getter/setter

Object.defineProperty 是 ES5 中一个无法 shim 的特性,这也就是 Vue 不支持 IE8 以及更低版本浏览器的原因。

这些 getter/setter 对用户来说是不可见的,但是在内部它们让 Vue 能够追踪依赖,在属性被访问和修改时通知变更。每个组件实例都对应一个 watcher 实例,它会在组件

渲染的过程中把“接触”过的数据属性记录为依赖。之后当依赖项的 setter 触发时,会通知 watcher,从而使它关联的组件重新渲染。

通过这么长时间的学习,我用比较直白的话来详细的解释一下原理

首先咱们需要掌握两个方法

一个是Object.defineProperty,一个是订阅者设计模式

Object.defineProperty方法

Object.defineProperty会直接在一个对象上定义一个新属性或者修改一个对象的现有属性,并且返回这个对象。

Object.defineProperty(obj,prop,descriptor)有三个属性。obj是要定义属性的对象。prop是需要定义或者修改的属性的名称,descrptor是被定义或修改的属性描述符。

setter或者getter是js对象中用来设置属性或获取属性的方法,他是在创建对象的时候指明的。

当data对象中有值存在时,vue就会获取这个对象,获取这个对象的时候就会调用get,然后会用Object.keys()方法拿到这个data对象中的每个属性,

然后进行遍历,得到每个属性,然后通过Object.defineProperty的set和get方法进行设置属性值或者是获取属性值。

发布-订阅模式

举个例子来说明一下发布者订阅者模式。上面的例子中,就是让张三和李四来订阅了这个message属性的改变。

发布-订阅是一种消息范式,消息的发送者(称为发布者)不会将消息直接发送给特定的接收者(称为订阅者)。而

是将发布的消息分为不同的类别,无需了解哪些订阅者(如果有的话)可能存在。同样的,订阅者可以表达对一个或多个类别的兴趣,

只接收感兴趣的消息,无需了解哪些发布者(如果有的话)存在。

举一个例子,你在微博上关注了A,同时其他很多人也关注了A,那么当A发布动态的时候,微博就会为你们推送这条动态。

A就是发布者,你是订阅者,微博就是调度中心,你和A是没有直接的消息往来的,全是通过微博来协调的(你的关注,A的发布动态)。

vue就是通过这种发布者订阅者模式来处理响应式数据的。

下面的代码只是把整个响应式的过程解释了一下,但是源码并不是这么写的,但是整个流程是这样走滴,只不过我简化可很多~~希望大家能明白哈

  // 他们全用了message
  <div>{{message}}</div>
    <p>{{message}}</p>
    <i>{{message}}</i>
 
   data() {
      return {
      message:'嘻嘻',
        name:'why'

    }

 
    },       
 
      // data传进了new Vue里面。内部拿到了data'中的对象,嗯,就拿这个名字命名为obj把,        
       var obj={           
        message:'哈哈',     
          name:'why'          };
// 然后通过遍历obj这个对象拿到这个属性------
      // Object.keys(obj)返回的是一个数组形式的这个对象的属性
// key代表的是每个属性
Object.keys(obj).forEach(key => {
// 拿到这个属性
let value=obj[key]
Object.defineProperty(obj,key,{
        // 设置属性
set(newValue) {
// 可以在这个位置监听key的改变
value=newValue
  
        // dep.notify() 然后在类似这个地方通知订阅者发生改变
          <!--  发生改变的时候,需要获取到谁在用这个值,假如这里有俩个人在用张三和李四,此时就需要对每个用到message进行解析,
          根据解析html代码,来获取到哪些人在用这些属性,他在获取message里的
          值的时候,他会调用一下get方法,谁用这个message,谁都会调取一次get,到时候就能知道到底是谁调用了这个message,
          一旦newValue发生改变,就会通知这三个人,然后会通知这三个人,让这三个人把界面更新一下 get方法---自身的update方法
          这时候就需要用发布订阅者模式来监听,让这三个人订阅这个属性的改变-->
                  console.log('监听'+key+'改变'+':'+value)
},
get() {
            在类似这个的地方创建一个watch的对象。来获取每个订阅者
            // const wat1=new Watcher('涵涵');
             console.log('获取'+key+'对应的值')
            return value
}
})
})
// 他就会执行对象里的set属性,把他的名字设置成'hanhan'
obj.name='hanhan'

我们把多个订阅者对象添加到发布者里面,一旦值发生改变,发布者只要去调用了自己的notify,就会立马通知之前所有的订阅者,

订阅者是一个数组,遍历数组里的每个成员,数组里面每个成员都有watch,通知订阅者去更新自己的update。然后进行更新界面

          // 发布者
class Dep {
constructor() {
// 订阅了一个数组,用这个数组去记录所有的订阅者
this.subscription=[]
}
// 加入订阅者,加进去这个人,也就是张三李四,这两个人
addSub(watch) {
this.subscription.push(watch)
};
notify() {
this.subscription.forEach(item=>{
// 调用它自己的unpate去更新
item.update()
})
}
} // 观察者,用于监听观察,通过这个类创建对象,
// 订阅者
class Watcher {
constructor(name) {
this.name=name;
} update() {
// 把自己的内容进行更新
console.log(this.name+'发生update')
}
} // 实例一个dep对象
const dep=new Dep(); const wat1=new Watcher('张三');
dep.addSub(wat1) //张三就被订阅者放到了subscription的数组里面 const wat2=new Watcher('李四');
dep.addSub(wat1) //李四就被订阅者放到了subscription的数组里面 dep.notify()//这里定义notify,那么刚才的两个订阅者,就全被我通知到了
}

下面我用一张图对上面的内容进行一个总结吧,总结的比较肤浅,都是表面的知识,希望大家多多指点,一块进步哈。

observer主要是对data对象中的数据进行了劫持监听,利用Object.defineProperty,一个属性对应一个dep对象,name有一个dep对象,age有一个dep对象,

他们是一一对应的关系,每个dep对象里面都 有他的观察者对象,观察者1,观察者2...当属性值发生变化的时候就会去调用dep中的notify,通知watcher,利用watcher去更新视图。

当el传进complie里面时,主要是做了两件事

一是解析html,创建对应的watcher,放到对应的observer中的dep对象中去,具体怎么放得,上面有说哈,

二是他还会根据el中的内容初始化view,也就是解析咱们的{{message}},在界面中显示出  “嘻嘻”。

假如我们把name中的属性值改成了 '哈哈 ',那么observer中的Object.defineProperty立马会监听到值的改变,调用notify的方法,遍历watcher,进行update,然后更新视图,把 why 变为 “哈哈”

分享就到这里了哈,楼楼有说的不好的地方,希望大家多多指点哈~~

vue学习之深入响应式原理的更多相关文章

  1. 《Vue 进阶系列之响应式原理及实现》

    https://www.bilibili.com/video/av51444410/?p=5 https://github.com/amandakelake/blog/issues/63 https: ...

  2. Vue.js学习 Item12 – 内部响应式原理探究

    深入响应式原理 大部分的基础内容我们已经讲到了,现在讲点底层内容.Vue.js 最显著的一个功能是响应系统 —— 模型只是普通对象,修改它则更新视图.这让状态管理非常简单且直观,不过理解它的原理也很重 ...

  3. vue 数组 新增元素 响应式原理 7种方法

    1.问题 思考一个问题,以下代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8" ...

  4. Vue.js 技术揭秘(学习) 深入响应式原理 nextTick外传

    microTask  mutationObserve. promise.then macroTask setImmediate. messageChannnel.setTimeout.postMess ...

  5. vue进阶用法-深入响应式原理

    异步更新队列 当vue异步执行更新DOM时,只要观察到数据变化,vue经开启一个队列,并缓冲在同一时间循环中发生的所有数据改变.如果同一个watch被多次触发,只会一次推入到队列中.然后在下一个事件循 ...

  6. 【Vue源码学习】响应式原理探秘

    最近准备开启Vue的源码学习,并且每一个Vue的重要知识点都会记录下来.我们知道Vue的核心理念是数据驱动视图,所有操作都只需要在数据层做处理,不必关心视图层的操作.这里先来学习Vue的响应式原理,V ...

  7. Vue.js响应式原理

      写在前面 因为对Vue.js很感兴趣,而且平时工作的技术栈也是Vue.js,这几个月花了些时间研究学习了一下Vue.js源码,并做了总结与输出. 文章的原地址:answershuto/learnV ...

  8. 深入浅出Vue基于“依赖收集”的响应式原理(转)

    add by zhj: 文章写的很通俗易懂,明白了Object.defineProperty的用法 原文:https://zhuanlan.zhihu.com/p/29318017 每当问到VueJS ...

  9. 深入解析vue.js响应式原理与实现

    vue.js响应式原理解析与实现.angularjs是通过脏检查来实现数据监测以及页面更新渲染.之后,再接触了vue.js,当时也一度很好奇vue.js是如何监测数据更新并且重新渲染页面.vue.js ...

随机推荐

  1. P5369 [PKUSC2018]最大前缀和

    状态压缩 P5369 题意:求所有排列下的最大前缀和之和 一步转化: 求最大前缀和的前缀由数集S组成的方案数, 统计答案时直接乘上sum(S)即可 考虑最大前缀和的性质: 设最大前缀和为sum[i] ...

  2. Java源码解析|String源码与常用方法

    String源码与常用方法 1.栗子 代码: public class JavaStringClass { public static void main(String[] args) { Strin ...

  3. office2019激活

    这个是在网上偶然看见的一个激活方式,分享一下. 复制如下代码保存后修改文件后缀名为".bat",请注意有一个点,然后保存以管理员身份运行即可: @echo off(cd /d &q ...

  4. 揭秘C# SQLite的从安装到使用

    SQLite,是一款轻型的数据库,是遵守ACID的关联式数据库管理系统,它的设计目标是嵌入式的,而且目前已经在很多嵌入式产品中使用了它,它占用资源非常的低,在嵌入式设备中,可能只需要几百K的内存就够了 ...

  5. [LeetCode] 704. Binary Search

    Description Given a sorted (in ascending order) integer array nums of n elements and a target value, ...

  6. Activity的四种加载模式详解:

    先来看看总结图: 模式详解: standard模式: 标准启动模式,也是activity的默认启动模式.在这种模式下启动的activity可以被多次实例化,即在同一个任务中可以存在多个activity ...

  7. 【DP合集】合并 union

    给出一个 1 ∼ N 的序列 A ( A 1 , A 2 , ..., A N ) .你每次可以将两个相邻的元素合并,合并后的元素权值即为 这两个元素的权值之和.求将 A 变为一个非降序列,最少需要多 ...

  8. java23种设计模式(一)工厂方法模式

    在说工厂方法模式之前,先了解一下简单工厂模式.工厂方法模式其实是在简单工厂上面做了一些增强. 简单工厂模式:有一个专门的类来生产其他类的实例,生产的这些实例有一个共同父类.这个跟我们的多态有一点像. ...

  9. Python:numpy中shape和reshape的用法

    >>> w=np.zeros((5,6))>>> warray([[ 0.,  0.,  0.,  0.,  0.,  0.],       [ 0.,  0.,  ...

  10. 网关服务自定义路由规则(springcloud+nacos)

    1. 场景描述 需要给各个网关服务类提供自定义配置路由规则,实时生效,不用重启网关(重启风险大),目前已实现,动态加载自定义路由文件,动态加载路由文件中的路由规则,只需在规则文件中配置下规则就可以了 ...