浅谈 vue 中的 watcher


Posted in Javascript onDecember 04, 2017

观察 Watchers

虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的 watcher 。这是为什么 Vue 提供一个更通用的方法通过watch 选项,来响应数据的变化。当你想要在数据变化响应时,执行异步操作或开销较大的操作,这是很有用的。

大家对于 watch 应该不陌生,项目中都用过下面这种写法:

watch: {
 someProp () {
  // do something
 }
}
// 或者
watch: {
 someProp: {
  deep: true,
  handler () {
   // do something
  }
 }
}

上面的写法告诉 vue,我需要监听 someProp 属性的变化,于是 vue 在内部就会为我们创建一个 watcher 对象。(限于篇幅,我们不聊 watcher 的具体实现,感兴趣的可以直接看源码 watcher)

然而在 vue 中,watcher 的功能并没有这么单一,先上段代码:

<template>
 <div>
  <p>a: {{ a }}</p>
  <p>b: {{ b }}</p>
  <button @click="increment">+</button>
 </div>
</template>
<script>
export default {
 data () {
  return {
   a: 1
  }
 },
 computed: {
  b () {
   return this.a * 2
  }
 },
 watch: {
  a () {
    console.log('a is changed')
  }
 },
 methods: {
  increment () {
   this.a += 1
  }
 },
 created () {
  console.log(this._watchers)
 }
}
</script>

在线demo

上面代码非常简单,我们现在主要关注 created 钩子中打印的 this._watchers,如下:

分别展开三个 watcher,观察每一个 expression,从上到下分别为:

b() {   return this.a * 2;↵  }
"a"
function () {   vm._update(vm._render(), hydrating);↵  }

上面三个 watcher 代表了三种不同功能的 watcher,我们将其按功能分为三类:

  • 在 watch 中定义的,用于监听属性变化的 watcher (第二个)
  • 用于 computed 属性的 watcher (第一个)
  • 用于页面更新的 watcher (第三个)

normal-watcher

我们在 watch 中定义的,都属于这种类型,即只要监听的属性改变了,都会触发定义好的回调函数

computed-watcher

每一个 computed 属性,最后都会生成一个对应的 watcher 对象,但是这类 watcher 有个特点,我们拿上面的 b 举例:

属性 b 依赖 a,当 a 改变的时候,b 并不会立即重新计算,只有之后其他地方需要读取 b 的时候,它才会真正计算,即具备 lazy(懒计算)特性

render-watcher

每一个组件都会有一个 render-watcher, function () {↵ vm._update(vm._render(), hydrating);↵ }, 当 data/computed

中的属性改变的时候,会调用该 render-watcher 来更新组件的视图

三种 watcher 的执行顺序

除了功能上的区别,这三种 watcher 也有固定的执行顺序,分别是:

computed-render -> normal-watcher -> render-watcher

这样安排是有原因的,这样就能尽可能的保证,在更新组件视图的时候,computed 属性已经是最新值了,如果 render-watcher 排在 computed-render 前面,就会导致页面更新的时候 computed 值为旧数据。

下面从一段实例代码中看下vue中的watcher

在这个示例中,使用 watch 选项允许我们执行异步操作(访问一个 API),限制我们执行该操作的频率,并在我们得到最终结果前,设置中间状态。这是计算属性无法做到的。

<div id="watch-example">
<p>
Ask a yes/no question:
<input v-model="question">
</p>
<p>{{ answer }}</p>
</div>
<!-- Since there is already a rich ecosystem of ajax libraries -->
<!-- and collections of general-purpose utility methods, Vue core -->
<!-- is able to remain small by not reinventing them. This also -->
<!-- gives you the freedom to just use what you're familiar with. -->
<script src="https://unpkg.com/axios@0.12.0/dist/axios.min.js"></script>
<script src="https://unpkg.com/lodash@4.13.1/lodash.min.js"></script>
<script>
var watchExampleVM = new Vue({
el: '#watch-example',
data: {
question: '',
answer: 'I cannot give you an answer until you ask a question!'
},
watch: {
 // 如果 question 发生改变,这个函数就会运行
question: function (newQuestion) {
this.answer = 'Waiting for you to stop typing...'
this.getAnswer()
}
},
methods: {
 // _.debounce 是一个通过 lodash 限制操作频率的函数。
 // 在这个例子中,我们希望限制访问yesno.wtf/api的频率
 // ajax请求直到用户输入完毕才会发出
 // 学习更多关于 _.debounce function (and its cousin
// _.throttle), 参考: https://lodash.com/docs#debounce
getAnswer: _.debounce(
function () {
var vm = this
if (this.question.indexOf('?') === -1) {
vm.answer = 'Questions usually contain a question mark. ;-)'
return
}
vm.answer = 'Thinking...'
axios.get('https://yesno.wtf/api')
.then(function (response) {
vm.answer = _.capitalize(response.data.answer)
})
.catch(function (error) {
vm.answer = 'Error! Could not reach the API. ' + error
})
},
// 这是我们为用户停止输入等待的毫秒数
500
)
}
})
</script>

小结

本文并不是源码解析类文章,只是从一个角度来聊聊,那些看似不相关的东西(computed/watch/页面更新),内部却有着紧密的联系,希望能抛砖引玉,让大家更深入的去探索 vue

Javascript 相关文章推荐
图片之间的切换
Jun 26 Javascript
JavaScript Base64编码和解码,实现URL参数传递。
Sep 18 Javascript
禁止iframe脚本弹出的窗口覆盖了父窗口的方法
Sep 06 Javascript
js拆分字符串并将分割的数据放到数组中的方法
May 06 Javascript
js实现鼠标移到链接文字弹出一个提示层的方法
May 11 Javascript
AngularJS中的过滤器filter用法完全解析
Apr 22 Javascript
jQuery实现点击后高亮背景固定显示的菜单效果【附demo源码下载】
Sep 21 Javascript
javascript中一些奇葩的日期换算方法总结
Nov 14 Javascript
vue登录注册实例详解
Sep 14 Javascript
vue + elementUI实现省市县三级联动的方法示例
Oct 29 Javascript
使用JS location实现搜索框历史记录功能
Dec 23 Javascript
Vue实现购物车基本功能
Nov 08 Javascript
vue中的计算属性的使用和vue实例的方法示例
Dec 04 #Javascript
Vue导出json数据到Excel电子表格的示例
Dec 04 #Javascript
微信小程序tabBar用法实例详解
Dec 04 #Javascript
详解如何实现一个简单的Node.js脚手架
Dec 04 #Javascript
浅谈React和Redux的连接react-redux
Dec 04 #Javascript
bootstrap3中container与container_fluid外层容器的区别讲解
Dec 04 #Javascript
深入浅出webpack之externals的使用
Dec 04 #Javascript
You might like
PHP的面试题集,附我的答案和分析(一)
2006/11/19 PHP
解决控件遮挡问题:关于有窗口元素和无窗口元素
2007/01/28 PHP
测试php函数的方法
2013/11/13 PHP
适用于初学者的简易PHP文件上传类
2015/10/29 PHP
PHP接收json 并将接收数据插入数据库的实现代码
2015/12/01 PHP
php正则表达式验证(邮件地址、Url地址、电话号码、邮政编码)
2016/03/14 PHP
PHP设计模式之单例模式原理与实现方法分析
2018/04/25 PHP
求得div 下 img的src地址的js代码
2007/02/28 Javascript
对JavaScript的eval()中使用函数的进一步讨论
2008/07/26 Javascript
Jquery实现点击切换图片并隐藏显示内容(2种方法实现)
2013/04/11 Javascript
jQuery实现跟随鼠标运动图层效果的方法
2015/02/02 Javascript
JavaScript使用简单正则表达式的数据验证功能示例
2017/01/13 Javascript
js验证手机号、密码、短信验证码代码工具类
2020/06/24 Javascript
微信小程序 支付功能开发错误总结
2017/02/21 Javascript
微信小程序页面跳转功能之从列表的item项跳转到下一个页面的方法
2017/11/27 Javascript
js实时监控文本框输入字数的实例代码
2018/01/18 Javascript
JavaScript canvas仿代码流瀑布
2020/02/10 Javascript
ant design 日期格式化的实现
2020/10/27 Javascript
python 排序算法总结及实例详解
2016/09/28 Python
Python管理Windows服务小脚本
2018/03/12 Python
Python自然语言处理 NLTK 库用法入门教程【经典】
2018/06/26 Python
python粘包问题及socket套接字编程详解
2019/06/29 Python
tensorflow之并行读入数据详解
2020/02/05 Python
Python搭建Keras CNN模型破解网站验证码的实现
2020/04/07 Python
Pycharm在指定目录下生成文件和删除文件的实现
2020/12/28 Python
利用python查看数组中的所有元素是否相同
2021/01/08 Python
美国知名的女性服饰品牌:LOFT(洛芙特)
2016/08/05 全球购物
英国知名小木屋定制网站:Tiger Sheds
2020/03/06 全球购物
运动会开幕式解说词
2014/02/05 职场文书
初中毕业生的自我评价
2014/03/03 职场文书
淘宝店策划方案
2014/06/07 职场文书
任命书标准格式
2015/03/02 职场文书
网吧员工管理制度
2015/08/05 职场文书
如何书写先进事迹材料?
2019/07/02 职场文书
中学生打架《检讨书》范文
2019/08/12 职场文书
教你如何用python开发一款数字推盘小游戏
2021/04/14 Python