详解Vue中watch的高级用法


Posted in Javascript onMay 02, 2018

假设有如下代码:

<div>
   <p>FullName: {{fullName}}</p>
   <p>FirstName: <input type="text" v-model="firstName"></p>
</div>

new Vue({
 el: '#root',
 data: {
  firstName: 'Dawei',
  lastName: 'Lou',
  fullName: ''
 },
 watch: {
  firstName(newName, oldName) {
   this.fullName = newName + ' ' + this.lastName;
  }
 } 
})

上面的代码的效果是,当我们输入firstName后,wacth监听每次修改变化的新值,然后计算输出fullName。

handler方法和immediate属性

这里 watch 的一个特点是,最初绑定的时候是不会执行的,要等到 firstName 改变时才执行监听计算。那我们想要一开始就让他最初绑定的时候就执行改怎么办呢?我们需要修改一下我们的 watch 写法,修改过后的 watch 代码如下:

watch: {
 firstName: {
  handler(newName, oldName) {
   this.fullName = newName + ' ' + this.lastName;
  },
  // 代表在wacth里声明了firstName这个方法之后立即先去执行handler方法
  immediate: true
 }
}

注意到handler了吗,我们给 firstName 绑定了一个handler方法,之前我们写的 watch 方法其实默认写的就是这个handler,Vue.js会去处理这个逻辑,最终编译出来其实就是这个handler。

而immediate:true代表如果在 wacth 里声明了 firstName 之后,就会立即先去执行里面的handler方法,如果为 false就跟我们以前的效果一样,不会在绑定的时候就执行。

deep属性

watch 里面还有一个属性 deep,默认值是 false,代表是否深度监听,比如我们 data 里有一个obj属性:

<div>
   <p>obj.a: {{obj.a}}</p>
   <p>obj.a: <input type="text" v-model="obj.a"></p>
</div>

new Vue({
 el: '#root',
 data: {
  obj: {
   a: 123
  }
 },
 watch: {
  obj: {
   handler(newName, oldName) {
     console.log('obj.a changed');
   },
   immediate: true
  }
 } 
})

当我们在在输入框中输入数据视图改变obj.a的值时,我们发现是无效的。受现代 JavaScript 的限制 (以及废弃 Object.observe),Vue 不能检测到对象属性的添加或删除。由于 Vue 会在初始化实例时对属性执行 getter/setter 转化过程,所以属性必须在 data 对象上存在才能让 Vue 转换它,这样才能让它是响应的。

默认情况下 handler 只监听obj这个属性它的引用的变化,我们只有给obj赋值的时候它才会监听到,比如我们在 mounted事件钩子函数中对obj进行重新赋值:

mounted: {
 this.obj = {
  a: '456'
 }
}

这样我们的 handler 才会执行,打印obj.a changed。

相反,如果我们需要监听obj里的属性a的值呢?这时候deep属性就派上用场了!

watch: {
 obj: {
  handler(newName, oldName) {
   console.log('obj.a changed');
  },
  immediate: true,
  deep: true
 }
}

deep的意思就是深入观察,监听器会一层层的往下遍历,给对象的所有属性都加上这个监听器,但是这样性能开销就会非常大了,任何修改obj里面任何一个属性都会触发这个监听器里的 handler。

优化,我们可以是使用字符串形式监听。

watch: {
 'obj.a': {
  handler(newName, oldName) {
   console.log('obj.a changed');
  },
  immediate: true,
  // deep: true
 }
}

这样Vue.js才会一层一层解析下去,直到遇到属性a,然后才给a设置监听函数。

Javascript 相关文章推荐
解析ScrollPic在ie8下只滚动一遍,然后变为空白 ie6,ie7,chrome,firefox正常
Jun 26 Javascript
仿百度输入框智能提示的js代码
Aug 22 Javascript
基于javascript实现判断移动终端浏览器版本信息
Dec 09 Javascript
js禁止页面刷新与后退的方法
Jun 08 Javascript
JSON字符串和对象相互转换实例分析
Jun 16 Javascript
vue+vue-router转场动画的实例代码
Sep 01 Javascript
使用React手写一个对话框或模态框的方法示例
Apr 25 Javascript
详解VScode编辑器vue环境搭建所遇问题解决方案
Apr 26 Javascript
atom-design(Vue.js移动端组件库)手势组件使用教程
May 16 Javascript
Vue axios 将传递的json数据转为form data的例子
Oct 29 Javascript
vue项目中定义全局变量、函数的几种方法
Nov 08 Javascript
微信小程序静默登录的实现代码
Jan 08 Javascript
Vue.js中关于侦听器(watch)的高级用法示例
May 02 #Javascript
Vue SSR 组件加载问题
May 02 #Javascript
基于jquery实现左右上下移动效果
May 02 #jQuery
关于Vue在ie10下空白页的debug小结
May 02 #Javascript
解析Json字符串的三种方法日常常用
May 02 #Javascript
使用vue-cli创建项目的图文教程(新手入门篇)
May 02 #Javascript
vue中的模态对话框组件实现过程
May 01 #Javascript
You might like
解析phpstorm + xdebug 远程断点调试
2013/06/20 PHP
dedecms函数分享之获取某一栏目所有子栏目
2014/05/19 PHP
浅析ThinkPHP的模板输出功能
2014/07/01 PHP
PHP获取网页所有连接的方法(附demo源码下载)
2016/03/30 PHP
PHP安全下载文件的方法
2016/04/07 PHP
Laravel多用户认证系统示例详解
2018/03/13 PHP
PHP+fiddler抓包采集微信文章阅读数点赞数的思路详解
2019/12/20 PHP
Google AJAX 搜索 API实现代码
2010/11/17 Javascript
director.js实现前端路由使用实例
2015/02/03 Javascript
jquery获取节点名称
2015/04/26 Javascript
javascript之Array 数组对象详解
2016/06/07 Javascript
canvas绘制一个常用的emoji表情
2017/03/30 Javascript
JavaScript限定范围拖拽及自定义滚动条应用(3)
2017/05/17 Javascript
详解vue axios中文文档
2017/09/12 Javascript
vue 项目如何引入微信sdk接口的方法
2017/12/18 Javascript
JavaScript实现获取select下拉框中第一个值的方法
2018/02/06 Javascript
Vue数据双向绑定的深入探究
2018/11/27 Javascript
在Python的web框架中编写创建日志的程序的教程
2015/04/30 Python
浅谈python str.format与制表符\t关于中文对齐的细节问题
2019/01/14 Python
Django stark组件使用及原理详解
2019/08/22 Python
python生成特定分布数的实例
2019/12/05 Python
Python使用jupyter notebook查看ipynb文件过程解析
2020/06/02 Python
Python2及Python3如何实现兼容切换
2020/09/01 Python
Python制作运行进度条的实现效果(代码运行不无聊)
2021/02/24 Python
NYX Professional Makeup英国官网:美国平价专业彩妆品牌
2019/11/13 全球购物
银行服务感言
2014/03/01 职场文书
幼儿园评语大全
2014/04/17 职场文书
敬老院院长事迹材料
2014/05/21 职场文书
慰问信模板
2015/02/14 职场文书
英语通知范文
2015/04/22 职场文书
教师见习总结范文
2015/06/23 职场文书
贴吧吧主申请感言
2015/08/03 职场文书
公司岗位说明书
2015/10/08 职场文书
小学学习委员竞选稿
2015/11/20 职场文书
MySQL表字段时间设置默认值
2021/05/13 MySQL
深入理解以DEBUG方式线程的底层运行原理
2021/06/21 Java/Android