vue使用watch监听属性变化


Posted in Vue.js onApril 30, 2022

Vue中可以使用监听器监听属性的变化,并根据属性变化作出响应。但一旦涉及到复杂数据的监听(如Object,但数组一般不需要,因为Vue针对数组做了特殊处理)时就比较复杂了,本文解释了使用watch监听属性变化的方法,包括复杂数据。

基本用法

Vue watch最重要的使用场景是根据某属性的变化执行某些业务逻辑:

<template>
  <input type="number" v-model.number="counter" />
</template>
<script>
export default {
  name: "Counter",
  data: function() {
    return {
      counter: 0,
    };
  },
  watch: {
    counter: function(newV, oldV) {
      console.log('counter change to %d from %d', newV, oldV);
    },
  }
};
</script>

watch的基本用法很简单:针对需要监听的属性定义个同名的函数即可,函数的第一个参数为变化后的值,第二个参数为变化前的值。

监听object

首先我们回顾一个JavaScript中的概念:复杂数据变量。“复杂”的原因在于变量只是一个引用,和C++中的指针类似,其保存的不是真实的数据,而是数据的地址。

比如对于一个object变量来说,添加属性、删除属性、修改属性的值都不会改变这个地址,这也可以说这个object变量没有变化。

不管所用的框架如何,基本定理肯定是生效的,所以Vue中监听object也是一难题,特别是嵌套数据的监听。

这里的变化指的是地址的变化,能够触发变化最简单的方式就是重新赋值。

<template>
  <div>
    <label>up trigger {{ counter.up }} times</label>
    <button @click="onTrigger('up')">Trigger Up</button>
    <br>
    <label>down trigger {{ counter.down }} times</label>
    <button @click="onTrigger('down')">Trigger down</button>
  </div>
</template>
<script>
export default {
  name: "Counter",
  data: function() {
    return {
      counter: {
        up: 0,
        down: 0,
      },
    };
  },
  methods: {
    onTrigger: function(type) {
      this.counter[type] += 1;
    }
  },
  watch: {
    counter: function(newV, oldV) {
      // 不会被触发
      console.log('counter change to %o from %o', newV, oldV);
    },
  }
};
</script>

针对counter的监听不会被触发,因为this.counter[type] += 1;并不会使this.counter变化(地址没变)。那如果想要监听到这个变化应该怎么办呢?一般来说有两种方式:

使用deep参数

watch: {
  counter: {
    handler: function(newV, oldV) {
      console.log('counter change to %o from %o', newV, oldV);
    },
    deep: true,
  }
}

使用deep需要使用watch的完整形式:handler是监听回调函数,deep: true指定了不仅仅监听counter的变化,也监听其内部属性的变化,所以当counter.up或counter.down变化时才能出发handler回调。

重新赋值

methods: {
  onTrigger: function(type) {
    // 重新赋值触发变化
    this.counter = {
      ...this.counter,
      [type]: this.counter[type] + 1,
    };
  }
},
watch: {
  counter: function(newV, oldV) {
    // 不会被触发
    console.log('counter change to %o from %o', newV, oldV);
  },
}

那两种方式优劣如何呢?使用deep参数会为数据每一层都添加监听,当层级较深时比较耗费性能,而且Vue不能监听到属性的添加或删除

所以一般来说使用重新赋值的方式是较优的方案,但如果只是想监听内部嵌

套数据的话,重新赋值就比较重了,所以Vue也提供了直接监听嵌套属性变化的途径:

通过路径监听内部数据

watch: {
  'counter.up': function(newV, oldV) {
    console.log('counter.up change to %d from %d', newV, oldV);
  },
  'counter.down': function(newV, oldV) {
    console.log('counter.down change to %d from %d', newV, oldV);
  },
}

通过这种方式可以避免使用deep造成的性能消耗问题,当只对某内部属性变化作出响应的场景下比较合适,但仍要注意监听的路径数据仍是复杂数据时的场景。

初始化变量触发监听回调

使用watch监听变化时,给变量初始值不会触发监听函数,如果像要改变这个默认设定可以使用immediate参数,其用法和deep类似:

watch: {
  counter: {
    handler: function(newV, oldV) {
      console.log('counter change to %o from %o', newV, oldV);
    },
    immediate: true,
  }
}

这样在赋初值时就会触发监听函数,其中第一个参数为初始值,第二个参数为undefined。

总结

使用watch可以监听属性的变化,且其使用方式也不少,理解每种方式的使用场景能为开发节省时间,优化性能。

watch使用文档

Vue Reactivity原理

computed vs watch


Tags in this post...

Vue.js 相关文章推荐
Vue.js桌面端自定义滚动条组件之美化滚动条VScroll
Dec 01 Vue.js
对vue生命周期的深入理解
Dec 03 Vue.js
Vue实现一种简单的无限循环滚动动画的示例
Jan 10 Vue.js
vue实现简易计算器功能
Jan 20 Vue.js
Vue仿Bibibili首页的问题
Jan 21 Vue.js
vue实现轮播图帧率播放
Jan 26 Vue.js
如何使用vue3打造一个物料库
May 08 Vue.js
Vue提供的三种调试方式你知道吗
Jan 18 Vue.js
VUE使用draggable实现组件拖拽
Apr 06 Vue.js
vue二维数组循环嵌套方式 循环数组、循环嵌套数组
Apr 24 Vue.js
vue.js 使用原生js实现轮播图
Apr 26 Vue.js
vue使用element-ui按需引入
May 20 Vue.js
vue-cli3.x配置全局的scss的时候报错问题及解决
vue项目如何打包之项目打包优化(让打包的js文件变小)
关于vue-router-link选择样式设置
Apr 30 #Vue.js
vue-treeselect的基本用法以及解决点击无法出现拉下菜单
Apr 30 #Vue.js
解决vue自定义组件@click点击失效问题
Apr 30 #Vue.js
Vue操作Storage本地化存储
Apr 29 #Vue.js
使用vuex-persistedstate本地存储vuex
Apr 29 #Vue.js
You might like
PHP实现json_decode不转义中文的方法
2017/05/20 PHP
PHP用户注册邮件激活账户的实现代码
2017/05/31 PHP
php设计模式之状态模式实例分析【星际争霸游戏案例】
2020/03/26 PHP
鼠标移动到一张图片时变为另一张图片
2006/12/05 Javascript
javascript将相对路径转绝对路径示例
2014/03/14 Javascript
JavaScript限定图片显示大小的方法
2015/03/11 Javascript
jQuery使用removeClass方法删除元素指定Class的方法
2015/03/26 Javascript
JS实现图片剪裁并预览效果
2016/08/12 Javascript
基于BootStrap与jQuery.validate实现表单提交校验功能
2016/12/22 Javascript
JavaScript 栈的详解及实例代码
2017/01/22 Javascript
jQuery.cookie.js实现记录最近浏览过的商品功能示例
2017/01/23 Javascript
关于JavaScript的单双引号嵌套问题
2017/08/20 Javascript
Vue实现typeahead组件功能(非常靠谱)
2017/08/26 Javascript
Vue2.0+ElementUI实现表格翻页的实例
2017/10/23 Javascript
vue bus全局事件中心简单Demo详解
2018/02/26 Javascript
bootstrap 路径导航 分页 进度条的实例代码
2018/08/06 Javascript
微信小程序身份证验证方法实现详解
2019/06/28 Javascript
解决Vue中使用keepAlive不缓存问题
2020/08/04 Javascript
DWR内存兼容及无法调用问题解决方案
2020/10/16 Javascript
[01:38]DOTA2辉夜杯 欢乐的观众现场采访
2015/12/26 DOTA
python爬虫实例详解
2018/06/19 Python
详解python的四种内置数据结构
2019/03/19 Python
python3 深浅copy对比详解
2019/08/12 Python
django admin 自定义替换change页面模板的方法
2019/08/23 Python
Python3批量移动指定文件到指定文件夹方法示例
2019/09/02 Python
Python模块汇总(常用第三方库)
2019/10/07 Python
Win10下python 2.7与python 3.7双环境安装教程图解
2019/10/12 Python
python爬虫添加请求头代码实例
2019/12/28 Python
Python使用ElementTree美化XML格式的操作
2020/03/06 Python
python3.8.1+selenium实现登录滑块验证功能
2020/05/22 Python
使用python把xmind转换成excel测试用例的实现代码
2020/10/12 Python
Sperry澳大利亚官网:源自美国帆船鞋创始品牌
2019/07/29 全球购物
大整数数相乘的问题
2012/07/22 面试题
维德科技C#面试题笔试题
2015/12/09 面试题
总结Python连接CS2000的详细步骤
2021/06/23 Python
Vue的生命周期一起来看看
2022/02/24 Vue.js