详解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 相关文章推荐
jQuery 剧场版 你必须知道的javascript
May 27 Javascript
javascript web页面刷新的方法收集
Jul 02 Javascript
javascript中类的定义及其方式(《javascript高级程序设计》学习笔记)
Jul 04 Javascript
js实现一个省市区三级联动选择框代码分享
Mar 06 Javascript
利用a标签自动解析URL分析网址实例
Oct 20 Javascript
JavaScript中操作字符串小结
May 04 Javascript
javascript+HTML5自定义元素播放焦点图动画
Feb 21 Javascript
node模块机制与异步处理详解
Mar 13 Javascript
AngularJS ng-change 指令的详解及简单实例
Jul 30 Javascript
vue中的scope使用详解
Oct 29 Javascript
swiper自定义分页器的样式
Sep 14 Javascript
vue 使用rules对表单字段进行校验的步骤
Dec 25 Vue.js
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
php file_exists 检查文件或目录是否存在的函数
2010/05/10 PHP
ThinkPHP的Widget扩展实例
2014/06/19 PHP
jQuery 1.5最新版本的改进细节分析
2011/01/19 Javascript
解决Extjs 4 Panel作为Window组件的子组件时出现双重边框问题
2013/01/11 Javascript
jQuery当鼠标悬停时放大图片的效果实例
2013/07/03 Javascript
Document:getElementsByName()使用方法及示例
2013/10/28 Javascript
基于jquery实现的可编辑下拉框实现代码
2014/08/02 Javascript
JSON.stringify转换JSON时日期时间不准确的解决方法
2014/08/08 Javascript
JavaScript中跨域调用Flash的方法
2014/08/11 Javascript
jquery实现页面百叶窗走马灯式翻滚显示效果的方法
2015/03/12 Javascript
Javascript监视变量变化的方法
2015/06/09 Javascript
深入浅出 jQuery中的事件机制
2016/08/23 Javascript
jQuery事件对象总结
2016/10/17 Javascript
jQuery中弹出iframe内嵌页面元素到父页面并全屏化的实例代码
2016/12/27 Javascript
jquery 实现复选框的全选操作实例代码
2017/01/24 Javascript
详解React服务端渲染从入门到精通
2019/03/28 Javascript
vue实现权限控制路由(vue-router 动态添加路由)
2019/11/04 Javascript
Vue项目打包压缩的实现(让页面更快响应)
2020/03/10 Javascript
JavaScript array常用方法代码实例详解
2020/09/02 Javascript
element-ui中el-upload多文件一次性上传的实现
2020/12/02 Javascript
[04:55]完美世界副总裁蔡玮:DOTA2的自由、公平与信任
2013/12/18 DOTA
Django中实现一个高性能计数器(Counter)实例
2014/07/09 Python
Python中使用OpenCV库来进行简单的气象学遥感影像计算
2016/02/19 Python
Python自定义进程池实例分析【生产者、消费者模型问题】
2016/09/19 Python
python逆向入门教程
2018/01/15 Python
python3第三方爬虫库BeautifulSoup4安装教程
2018/06/19 Python
python使用Plotly绘图工具绘制柱状图
2019/04/01 Python
python 批量将中文名转换为拼音
2021/02/07 Python
美体小铺英国官网:The Body Shop英国
2017/01/24 全球购物
马来西亚在线健康商店:Medipal Malaysia
2020/04/13 全球购物
教师中国梦演讲稿
2014/04/23 职场文书
建筑管理专业求职信
2014/07/28 职场文书
文明好少年事迹材料
2014/08/19 职场文书
优秀班主任工作总结2015
2015/05/25 职场文书
2015年秋学期师德师风建设工作总结
2015/10/23 职场文书
承诺书的内容有哪些,怎么写?
2019/06/21 职场文书