vue取通过key取value_彻底理解Vue中的Watcher、Observer、Dep

思考以下代码

new Vue({el: '#example',data(){return{obj:{a:1}}},
})

当我们写下这行代码时,vue将我们在data内定义的obj对象进行依赖追踪.

具体做法为执行new Observer(obj)

//经过上面的代码,我们的obj对象会变为以下的样子
{obj:{a:1,__ob__:{ //Observer 实例dep:{Dep 实例subs:[ //存放 Watcher 实例new Watcher(),]}}}
}

我们来一步步实现看下。 1. 在obj对象上新增__ob__属性,值为Observe 类的实例,我们编写一个 def 函数,用来增加属性

function def(obj, key, val, enumerable) {Object.defineProperty(obj, key, {value: val,enumerable: !!enumerable,writable: true,configurable: true});
}

增加啥属性呢?之前提到了,我们需要增加一个 Observer 实例,实现如下

Observe 实现

class Observer {constructor(targetObject) {def(targetObject, '__ob__', this);//在 targetObject 上 添加  Observer 实例, setter时 通知该实例this.walk(targetObject)this.dep = new Dep()}walk(obj) {Object.keys(obj).forEach(key => {defineReactive(obj, key, obj[key])});}}function observe(data) {if (Object.prototype.toString.call(data) !== '[object Object]') {return}new Observer(data)
}function defineReactive(obj, key, val) {observe(val)Object.defineProperty(obj, key, {enumerable: true,configurable: true,get: function reactiveGetter() {console.log('get');const ob = this.__ob__ob.dep.depend();return val},set: function reactiveSetter(newVal) {console.log('set');if (newVal === val) returnval = newValobserve(newVal)const ob = this.__ob__ob.dep.notify();},})
}function def(obj, key, val, enumerable) {Object.defineProperty(obj, key, {value: val,enumerable: !!enumerable,writable: true,configurable: true});
}

这里面牵扯到了 Dep,我们也把Dep实现下,

Dep

class Dep {constructor() {this.subs = []}addSub(sub) {this.subs.push(sub)}depend() {this.subs.push(Dep.target)}notify() {for (let i = 0; i < this.subs.length; i++) {this.subs[i].fn()}}
}Dep.target = null

Observer 类 主要做了以下事情

  1. 遍历 data 下的每一个属性,若是对象,则 执行 new Observer() ,在对象上新增__ob__属性,该属性的值为 Observer 的实例
  2. 劫持对象属性的变化,在 getter 的时候,拿到 Observer 实例的dep实例,执行dep.depend(),代码如下
const ob = this.__ob__ob.dep.depend();

看下 dep.depend()做了些啥

this.subs.push(Dep.target)

Dep.target添加到 订阅数组内(this.subs)

也就是说,只要我们 Dep.target 赋值了,再执行 dep.depend(),那么该值就会被添加到 dep 的 subs 数组内,比如
Dep.target =function test(){}
dep.depend()
// test 函数就算 Dep 的订阅者了

实现自动添加依赖

这个时候该 Watcher出场了

Watcher 实现

const Dep = require('./Dep')class Watcher {constructor(vm, exp, fn) {this.vm = vmthis.exp = expthis.fn = fnDep.target = this//将自己挂载到 Dep.target,调用 Dep.depend时会读取该变量this.vm[exp]}
}module.exports = Watcher

根据一个小例子来理解 Watcher

const obj = {a: 1,b: {c: 2}
}new Observer(obj)
new Watcher(obj, 'a', () => {console.log('Watcher 回调执行')
})
obj.a='222'

流程如下: 1. 先观测 obj 对象(new Observer(obj)) 2. 实例化Watcher时,会执行Dep.target = this,然后执行this.vm[exp],也就是取一次值,那么会触发 getter,将自身(Watcher实例)添加到dep的订阅者数组内

get: function reactiveGetter() {const ob = this.__ob__ob.dep.depend();return val},

最后,改变数据时候,触发setter

set: function reactiveSetter(newVal) {if (newVal === val) returnval = newValobserve(newVal)const ob = this.__ob__ob.dep.notify();},

执行ob.dep.notify()

notify() {for (let i = 0; i < this.subs.length; i++) {this.subs[i].fn()}

遍历 订阅者(subs)执行回调函数,整个流程结束

Leeesin/learn-vue-source-code​github.com
12c2ace689b2077fac48d8892af47976.png

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

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

相关文章

浏览器打开服务器上的图片无法显示,网页中的图片打不开怎么办?原因与解决办法...

最近有网友问小编这样一个很泛的问题&#xff1a;网页中的图片打不开怎么办&#xff1f;对于这个问题&#xff0c;其实导致的原因有很多&#xff0c;但也很好排除原因&#xff0c;主要从网络&#xff0c;网页&#xff0c;平台等当面综合去分析&#xff0c;就很容易可以找到答案…

WCF学习之旅—实现支持REST客户端应用(二十四)

WCF学习之旅—实现REST服务&#xff08;二十二&#xff09; WCF学习之旅—实现支持REST服务端应用&#xff08;二十三&#xff09; 在上二篇文章中简单介绍了一下RestFul与WCF支持RestFul所提供的方法&#xff0c;及创建一个支持REST的WCF服务端程序&#xff0c;本文介绍如何调…

arduino智能浇花系统_创新成果 | 养花神器——智能浇花机

养花的人应该多少都会遇到这样的问题&#xff1a;需要根据花的习性定时浇水&#xff0c;但给花浇水是个「技术活」&#xff0c;不记得浇水或水浇多了&#xff0c;都会影响花的生长。针对这个问题&#xff0c;可以研究制作一个简单、实用的自动浇花机。根据检测土壤的湿度&#…

导出怎么用_微信好友账号怎么导出?微信怎么备份通讯录?

微信好友账号怎么导出?微信怎么备份通讯录?对于经常使用微信社交的朋友&#xff0c;都有一个共同的烦恼&#xff0c;那就是微信只能加5000多个好友&#xff0c;平时还怕一个不小心误删了微信好友&#xff0c;不要烦恼&#xff0c;小编教你导出微信好友!情景一&#xff1a;登录…

cvpr 深度估计_CVPR再现黑科技!你还在相信“眼见为实”?

全文共2634字&#xff0c;预计学习时长7分钟经常听到有人说&#xff0c;我除了自己的眼睛&#xff0c;什么也不信。自从09年阿凡达(Avatar)电影上映以来&#xff0c;3D渲染、虚拟现实的逼真度总是让人叹为观止。而今&#xff0c;10年过去&#xff0c;最近计算机视觉领域顶级会议…

360浏览器查看服务器响应内容,360浏览器怎么看3个月以前的浏览记录?

360浏览器怎么看3个月以前的浏览记录&#xff1f;有时候我们想查找浏览器的浏览记录&#xff0c;这个比较简单&#xff0c;Ctrlh直接查看&#xff0c;或者如图直接查看。然而有些记录在3个月之前就坑爹了&#xff0c;浏览器是看不到的怎么办呢&#xff1f;比如几天是10月1号&am…

【Electron】Electron开发入门(一):开发环境搭建

刚接触Electronjs开发PC端桌面应用程序的时候&#xff0c;简直一头雾水&#xff0c;搜了网上很多教程&#xff0c;有的要么讲的零零碎碎&#xff0c;要么就是版本太低&#xff0c;很多API语法都不能用了&#xff1b;现在我把一些有用的教程归纳一下&#xff0c;并把目前最新Ele…

方差为什么用平方不用绝对值_为什么戚风蛋糕用玉米油而不用黄油?

刚开始做蛋糕的时候是最喜欢改配方的&#xff0c;认为糖太多了把糖减点&#xff0c;油太多了把油减点&#xff0c;黄油更香能不能把玉米油换成黄油&#xff0c;西点不是大都用黄油的&#xff1f;来&#xff0c;来&#xff01;我们来讨论下为什么戚风蛋糕要用玉米油而不用黄油&a…

中国服务器销售排名,IDC Q3:华为FusionServer Pro智能服务器发货量、销售额荣登中国区x86标准服务器排名双冠王...

据IDC发布的《中国区服务器季度跟踪报告》显示&#xff0c;2020年第三季度中国区x86标准服务器市场(标注1)&#xff0c;华为FusionServer pro智能服务器发货量及销售额双居中国区第一(标注2)&#xff0c;其中机架、高密型号服务器&#xff0c;均列单品类发货量、销售额第一&…

旅游流的概念_2020年去张家界凤凰古城旅游亲身体验经历分享——实用攻略(图文)...

张家界冬暖夏凉&#xff0c;一年四季都适合游玩&#xff0c;是国内旅游首选的休闲度假胜地&#xff0c;作为到过张家界几次的我&#xff0c;写了一份比较详细的旅游攻略&#xff0c;仅供参考&#xff01;不喜勿喷。我是靖宇&#xff0c;喜欢旅行和记录&#xff0c;每去一个地方…

服务器虚拟光驱无法加载,Proxmox/创建PVE/安装windows 2012r2系统无法识别硬盘/如何添加virtio驱动/...

前面一遍文章写了proxmox如何创建centos7系统的小鸡儿&#xff0c;那proxmox如何创建windows系统的小鸡儿呢&#xff1f;尤其是当我们小鸡的硬盘设置成virtio SCSI的时候&#xff0c;这时候windows ISO如果不包含virtio驱动&#xff0c;是无法识别硬盘&#xff0c;是无法装机的…

sudo: Cannot execute /usr/local/bin/zsh: No such file or directory 问题

参考&#xff1a;sudo: Cannot execute /usr/local/bin/zsh: No such file or directory 之前在美化Ubuntu的时候&#xff0c;下了个zsh&#xff0c;但是忘记改配置文件中的路径了&#xff0c;于是在su root的时候出现该情况。 解决方法&#xff1a; sudo vim /etc/passwd 将 r…

如何给对方邮箱发照片_朋友圈如何发心形拼图九宫格照片?

导读&#xff1a;经常会看到朋友圈里别人秀恩爱的爱心九宫格照片&#xff0c;此篇文章教你如何不用ps也可以制作心形拼图&#xff01;找了很多制作心形拼图的app&#xff0c;但都不是我在朋友圈看到别人发的那种心形制作的图&#xff0c;阅尽千帆&#xff0c;终于让我找到这种心…

触发起名字使用正则_好名字一定在字音、字形、字意上比较吉利

专注宝宝起名20年&#xff01;擅长结合生辰八字五行周易等综合起名&#xff0c;免费起名加专家微信 bbqm8888 (长按复制)起名字要说简单也简单&#xff0c;要说难确实也很难&#xff0c;主要取决于想如何起名字了。如果觉得名字只是一个代称&#xff0c;找几个自己喜欢的…

fanuc机器人四边形编程_中国工控 | FANUC 机器人码垛编程详解

(关注ID&#xff1a;chinak958888)(工控技术自媒体领导者)关注我们&#xff0c;结交自动化技术人中国工控技术学习媒体1. 码垛功能的定义对几个具有代表性的点进行示教&#xff0c;即可以从下层到上层按照顺序堆叠工件。2. 码垛的种类码垛 B:包括码垛B(单路径模式)和码垛BX(多路…

使用缓存的9大误区(上)(转)

如果说要对一个站点或者应用程序经常优化&#xff0c;可以说缓存的使用是最快也是效果最明显的方式。一般而言&#xff0c;我们会把一些常用的&#xff0c;或者需要花费大量的资源或时间而产生的数据缓存起来&#xff0c;使得后续的使用更加快速。 如果真要细说缓存的好处&…

cmake取消宏定义_Excel基础丨取消excel中宏安全提示框

很多高手都喜欢使用“宏”命令来提高办公工作效率。但在 Microsoft Office 程序中使用宏时&#xff0c;总会弹出宏安全警告&#xff0c;这让使用者倍感麻烦。而如果把宏的安全级设置为“低”&#xff0c;就可以取消excel中宏安全提示框了&#xff0c;又增加了恶意代码和病毒攻击…

5b计算机联锁系统_力控科技油库调度管理系统应用案例

一 、 项目概述"油库储油罐区具有分布空间范围广、安全防爆要求高、监控点多、布线复杂&#xff0c;自动化系统的水平和垂直集成难度大等特点。系统采用先进测控与管理技术&#xff0c;围绕储油罐区自动计量、监测与管理&#xff0c;进行储油罐区监测控制与数据采集系统的…

集合框架(九)----Map

从这篇开始就要学习Map了&#xff01; 先来看一下Map的继承体系&#xff1a; 如君所见&#xff0c;Maps的体系没有像Collections那么庞大 在接下来的两篇博文中将分别介绍HashMap和TreeMap 至于WeakHashMap,个人感觉文档中说的相当含糊&#xff0c;理解起来很有难度&#xff0c…

cpu java poi 导出_让 Java 开发更简单,提高工作效率 | Gitee 项目推荐

1、项目名称&#xff1a;基于 Spring Boot 的权限管理系统项目简介&#xff1a;Good 权限管理系统是作者学习 springBoot 时基于 springBoot 开发的一套轻量级的权限系统&#xff0c;其目的是形成一套属于自己的通用的开发框架 以后来项目的时候就可以直接基于此平台进行开发&a…