vue简版源码 Watcher
2021/5/5 20:25:50
本文主要是介绍vue简版源码 Watcher,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
vue简版源码 Watcher
<script src="./mvvm-master/js/watcher.js"></script> // watcher 订阅者
Watcher
function Watcher(vm, exp, cb) { //在watcher的实例上保存回调函数 this.cb = cb; //用于更新界面的回调函数 this.vm = vm; //MVVM的实例vm this.exp = exp; //对应的表达式 this.depIds = {}; //n个相关的dep的容器 this.value = this.get(); //初始化获取当前表达式对应的value } Watcher.prototype = { update: function() { this.run(); }, run: function() { var value = this.get(); var oldVal = this.value; if (value !== oldVal) { this.value = value; //调用构建watcher实例的时候传递过来的回调函数,并且确定他的this指向为vm this.cb.call(this.vm, value, oldVal); } }, //添加dep到watcher实例的depIds里 addDep: function(dep) { //判断当前depIds中是否已经拥有了dep //当模版首次更新的时候都是没有的,需要添加,当页面数据发生改变的时候同时会调用此方法,但是depIds中已经拥有了当前的dep,所以就不用再添加了 if (!this.depIds.hasOwnProperty(dep.id)) { //调用dep的addSub方法进行对watcher实例的添加 dep.addSub(this);//此时this是watcher的实例 //添加当前的dep到depIds中,并且把dep.id的值作为depIds的key this.depIds[dep.id] = dep; } }, get: function() { //在watcher初始化的时候保存实例到Dep的target上 Dep.target = this; //调用get方法获取当前对应表达式的数据 var value = this.getVMVal(); //清空target Dep.target = null; return value; }, getVMVal: function() { var exp = this.exp.split('.'); var val = this.vm._data; exp.forEach(function(k) { //读取_data中表达式所对应的值 (此时会触发observer中的get方法) val = val[k]; }); return val; } }; // 1. 每次调用run()的时候会触发相应属性的getter // getter里面会触发dep.depend(),继而触发这里的addDep // 2. 假如相应属性的dep.id已经在当前watcher的depIds里,说明不是一个新的属性,仅仅是改变了其值而已 // 则不需要将当前watcher添加到该属性的dep里 // 3. 假如相应属性是新的属性,则将当前watcher添加到新属性的dep里 // 如通过 vm.child = {name: 'a'} 改变了 child.name 的值,child.name 就是个新属性 // 则需要将当前watcher(child.name)加入到新的 child.name 的dep里 // 因为此时 child.name 是个新值,之前的 setter、dep 都已经失效,如果不把 watcher 加入到新的 child.name 的dep中 // 通过 child.name = xxx 赋值的时候,对应的 watcher 就收不到通知,等于失效了 // 4. 每个子属性的watcher在添加到子属性的dep的同时,也会添加到父属性的dep // 监听子属性的同时监听父属性的变更,这样,父属性改变时,子属性的watcher也能收到通知进行update // 这一步是在 this.get() --> this.getVMVal() 里面完成,forEach时会从父级开始取值,间接调用了它的getter // 触发了addDep(), 在整个forEach过程,当前wacher都会加入到每个父级过程属性的dep // 例如:当前watcher的是'child.child.name', 那么child, child.child, child.child.name这三个属性的dep都会加入当前watcher
这篇关于vue简版源码 Watcher的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2024-12-29如何在 Vue2 的 uni-app 项目中使用 npm ?-icode9专业技术文章分享
- 2024-12-29uni-app vue2微信小程序项目在哪里打开终端并使用npm?-icode9专业技术文章分享
- 2024-12-29怎么在 uni-app Vue2 项目中全局引入 Vant Weapp?-icode9专业技术文章分享
- 2024-12-29uni-app vue2微信小程序项目如何在main.js中全局引入vant?-icode9专业技术文章分享
- 2024-12-28Vue入门教程:从零开始搭建第一个Vue项目
- 2024-12-28Vue CLI入门指南:快速搭建Vue项目
- 2024-12-28Vue3基础知识入门教程
- 2024-12-28Vue3公共组件开发与使用入门教程
- 2024-12-28Vue CLI学习:新手入门教程
- 2024-12-28Vue CLI学习:轻松入门与实践指南