vue 源码自问自答-响应式原理

最近看了 Vue 源码和源码分析类的文章,感觉明白了很多,但是仔细想想却说不出个所以然。

所以打算把自己掌握的知识,试着组织成自己的语言表达出来

不打算平铺直叙的写清楚 vue 源码的前因后果和全部细节,而是以自问自答的形式,回答我自己之前的疑惑,

如果有错误的地方,欢迎指正哈~

Vue 的双向数据绑定原理

Vue 实现响应式的核心 API 是 ES5 的 Object.defineProperty(obj,key,descriptor),Vue 的「响应式」和「依赖收集」都依靠这个 API

它接受 3 个参数,分别是 obj / key / 描述符,返回的是一个包装后的对象

它的作用就是,用这个 API 包装过后的对象可以拥有 getter 和 setter 函数。

getter 会在对象的这个 key 被获取时触发,setter 会在这个对象的 key 被修改时触发。

一个 Vue 项目的开始, 通常是从 Vue 构造函数的实例化开始的。

new Vue()的时候会执行一个_init()方法,会初始化属性,比如 props/event/生命周期钩子,也包括 data 对象的初始化。

Vue 在初始化时,将 data 对象上的所有 key,都包装成拥有 getter 和 setter 的属性。

  • 渲染页面时,会执行 render function(无论是用 template 还是 render 最终都会生成 render 函数)
  • 执行 render function 会获取 data 对象上的属性,所以会触发对应属性的 getter 函数
  • getter 触发时,主要就做 2 个事情。 1.把值返回 2.依赖收集,也就是讲 watcher 存放到 Dep 实例的一个队列里。
  • 当修改对象的值触发 setter,setter 同样是做 2 个事情。1.把 newVal 设置好 2.用一个循环通知之前存放在 dep 实例中 watcher 对象们,watcher 对象调用各自的 update 方法来更新视图

实现双向数据绑定的demo1 - 忽略「收集依赖」的版本


function cb() {
console.log("更新视图");
} function defineReactve(obj, key, val) {
Object.defineProperty(obj, key, {
enumerable: true,
configurable: true,
get: () => {
console.log("触发了getter");
return val;
},
set: newVal => {
console.log("触发了setter");
if (newVal === val) return;
val = newVal;
cb()
}
});
} function observe(data) {
function walk(data) {
Object.keys(data).forEach(key => {
if (typeof data[key] === "object") {
walk(data[key]);
} else {
defineReactve(data, key, data[key]);
}
});
}
walk(data);
} class Vue {
constructor(options) {
this._data = options.data;
observe(this._data);
}
} var vm = new Vue({
data: {
msg: "test",
person: {
name: "ziwei",
age: 18
}
}
}); vm._data.person.name = 'hello' // 触发setter和cb函数,从而视图更新

实现双向数据绑定的demo2 - 「收集依赖」的版本


function defineReactve( obj, key, val ) {
const dep = new Dep()
Object.defineProperty( obj, key, {
enumerable: true,
configurable: true,
get: () => {
console.log( "触发了getter" );
dep.addSub(Dep.target)
return val;
},
set: newVal => {
console.log( "触发了setter" );
if ( newVal === val ) return;
val = newVal;
dep.notify() // 通知队列的wather去update视图
}
} );
} function observe( data ) {
function walk( data ) {
Object.keys( data ).forEach( key => {
if ( typeof data[ key ] === "object" ) {
walk( data[ key ] );
} else {
defineReactve( data, key, data[ key ] );
}
} );
}
walk( data );
} class Dep{
constructor(){
this.subs = []
} addSub(){
this.subs.push(Dep.target)
} notify(){
this.subs.forEach(sub => {
sub.update()
})
}
} Dep.target = null class Watcher{
constructor(){
Dep.target = this
} update(){
console.log('update更新视图啦~')
}
} class Vue {
constructor( options ) {
this._data = options.data;
observe( this._data ); new Watcher() // 模拟页面渲染,触发getter,依赖收集的效果
this._data.person.name
}
} var vm = new Vue( {
data: {
msg: "test",
person: {
name: "ziwei",
age: 18
}
}
} ); vm._data.person.name = 'hello'

一些省略掉的环节

这样就是 Vue 响应式的一个基本原理,不过我描述的过程中,也省略了很多环节,比如

  • Vue 是如何实现给 data 对象上的属性都拥有 getter 和 setter 的
  • 为什么要进行「依赖收集」,
  • 如何避免重复「收集依赖」
  • watcher 调用 update,也并不是直接更新视图。实现上中间还有 patch 的过程以及使用队列来异步更新的策略。

Vue 是如何实现给 data 对象上的属性都拥有 getter 和 setter 的


通过循环data对象,给对象的每一个key,用Object.defineProperty包装 遍历时,如果发现data[key]也是对象的话,需要用递归

为什么要进行「依赖收集」?

举2个场景的栗子

vue 源码自问自答-响应式原理的更多相关文章

  1. Vue 源码解析:深入响应式原理(上)

    原文链接:http://www.imooc.com/article/14466 Vue.js 最显著的功能就是响应式系统,它是一个典型的 MVVM 框架,模型(Model)只是普通的 JavaScri ...

  2. java官网门户源码 SSM框架 自适应-响应式 freemarker 静态模版引擎

    来源:http://www.fhadmin.org/webnewsdetail3.html 前台:支持(5+1[时尚单页风格])六套模版,可以在后台切换 官网:www.fhadmin.org 系统介绍 ...

  3. Vue2.0源码阅读笔记(二):响应式原理

      Vue是数据驱动的框架,在修改数据时,视图会进行更新.数据响应式系统使得状态管理变的简单直接,在开发过程中减少与DOM元素的接触.而深入学习其中的原理十分有必要,能够回避一些常见的问题,使开发变的 ...

  4. 手摸手带你理解Vue响应式原理

    前言 响应式原理作为 Vue 的核心,使用数据劫持实现数据驱动视图.在面试中是经常考查的知识点,也是面试加分项. 本文将会循序渐进的解析响应式原理的工作流程,主要以下面结构进行: 分析主要成员,了解它 ...

  5. vue响应式原理,去掉优化,只看核心

    Vue响应式原理 作为写业务的码农,几乎不必知道原理.但是当你去找工作的时候,可是需要造原子弹的,什么都得知道一些才行.所以找工作之前可以先复习下,只要是关于vue的,必定会问响应式原理. 核心: / ...

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

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

  7. Vue 源码解读(3)—— 响应式原理

    前言 上一篇文章 Vue 源码解读(2)-- Vue 初始化过程 详细讲解了 Vue 的初始化过程,明白了 new Vue(options) 都做了什么,其中关于 数据响应式 的实现用一句话简单的带过 ...

  8. 学习 vue 源码 -- 响应式原理

    概述 由于刚开始学习 vue 源码,而且水平有限,有理解或表述的不对的地方,还请不吝指教. vue 主要通过 Watcher.Dep 和 Observer 三个类来实现响应式视图.另外还有一个 sch ...

  9. vue源码之响应式数据

    分析vue是如何实现数据响应的. 前记 现在回顾一下看数据响应的原因. 之前看了vuex和vue-i18n的源码, 他们都有自己内部的vm, 也就是vue实例. 使用的都是vue的响应式数据特性及$w ...

随机推荐

  1. 用hdparm获取硬盘参数

    hdparm是Linux下一款能够获取和设置SATA/IDE设备参数的工具. 1.获取硬盘参数 $ sudo hdparm -i /dev/sda$ sudo hdparam -i /dev/sda ...

  2. iOS 监测电话呼入

    1.首先引入CoreTelephony框架,代码里: @import CoreTelephony; 项目设置里: 2.定义属性,建立强引用: @property (nonatomic, strong) ...

  3. Oracle 查询约束信息

    select * from user_constraints where table_name=''; select * from user_cons_columns;

  4. the little schemer 笔记(4)

    第四章 numbers games 14 是原子吗 是的,数都是原子 (atom? n) 是真还是假,其中n是14 真,14 是原子 -3是数吗 是的,不过我们暂不考虑负数 3.14159是数吗 是的 ...

  5. nginx使用autoindex

    有时候一个nginx服务就是为了用来下载文件的,网上很多下载服务都是这样的 这个很简单 在http段加上以下参数,重启nginx就行. autoindex on; autoindex_exact_si ...

  6. 微信小程序资料收集(一)

    1.微信小程序用户授权 https://blog.csdn.net/qq_34827048/article/details/77990510 https://blog.csdn.net/qq_3361 ...

  7. Python实现两已知排好序的列表合并成一个排好序的列表

    #方法0.5--- lst1 = [1, 3, 7, 9, 12] lst2 = [4, 8, 9, 13, 15, 19] def merge(a, b): c = [] h = j = 0 whi ...

  8. 《统计学习方法》笔记三 k近邻法

    本系列笔记内容参考来源为李航<统计学习方法> k近邻是一种基本分类与回归方法,书中只讨论分类情况.输入为实例的特征向量,输出为实例的类别.k值的选择.距离度量及分类决策规则是k近邻法的三个 ...

  9. C#将excel数据按照需求导入Sql server遇到的问题(参考而已)

    1.千万不要使用永中表格(WPS没用过,这里只是个人观点,不是说永中表格的) 我在公司得到的任务是将excel数据按照需求导入数据库总共主表大概3张,所以有点复杂(列子用的简单表,公司东西还是不要放出 ...

  10. Load average in Linux的精确含义

    Man 上的解释: load average System load averages is the average number of processes that are either in a ...