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中通过render函数给子组件设置ref操作
Nov 17 Vue.js
vue-drawer-layout实现手势滑出菜单栏
Nov 19 Vue.js
在Vue中使用mockjs代码实例
Nov 25 Vue.js
vue中如何自定义右键菜单详解
Dec 08 Vue.js
vue 递归组件的简单使用示例
Jan 14 Vue.js
Vue 3自定义指令开发的相关总结
Jan 29 Vue.js
vue如何使用rem适配
Feb 06 Vue.js
vue项目两种方式实现竖向表格的思路分析
Apr 28 Vue.js
详解Vue项目的打包方式(生成dist文件)
Jan 18 Vue.js
vue实现在data里引入相对路径
Jun 05 Vue.js
VUE递归树形实现多级列表
Jul 15 Vue.js
Vue3实现简易音乐播放器组件
Aug 14 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开发
2015/09/28 PHP
详解WordPress中调用评论模板和循环输出评论的PHP函数
2016/01/05 PHP
使用PHP处理数据库数据如何将数据返回客户端并显示当前状态
2016/02/16 PHP
PHP的new static和new self的区别与使用
2019/11/27 PHP
5款Javascript颜色选择器
2009/10/25 Javascript
js TextArea的选中区域处理
2010/12/28 Javascript
javascript解决innerText浏览器兼容问题思路代码
2013/05/17 Javascript
JavaScript对象学习经验整理
2013/10/12 Javascript
Javascript前端UI框架Kit使用指南之kitjs的对话框组件
2014/11/28 Javascript
a标签跳转到指定div,jquery添加和移除class属性的实现方法
2016/10/10 Javascript
Jquery Easyui自定义下拉框组件使用详解(21)
2020/12/31 Javascript
JS实现改变HTML上文字颜色和内容的方法
2016/12/30 Javascript
layui前段框架日期控件使用方法详解
2017/05/19 Javascript
JS判断一个数是否是水仙花数
2017/06/11 Javascript
利用vue.js实现被选中状态的改变方法
2018/02/08 Javascript
JQueryDOM之样式操作
2019/03/27 jQuery
在Node.js中将SVG图像转换为PNG,JPEG,TIFF,WEBP和HEIF格式的方法
2019/08/22 Javascript
js实现贪吃蛇游戏(简易版)
2020/09/29 Javascript
Python中实现结构相似的函数调用方法
2015/03/10 Python
python使用append合并两个数组的方法
2015/04/28 Python
Python开发微信公众平台的方法详解【基于weixin-knife】
2017/07/08 Python
python3.6+opencv3.4实现鼠标交互查看图片像素
2018/02/26 Python
Python判断两个list是否是父子集关系的实例
2018/05/04 Python
Pytorch对Himmelblau函数的优化详解
2020/02/29 Python
基于python实现上传文件到OSS代码实例
2020/05/09 Python
20行代码教你用python给证件照换底色的方法示例
2021/02/05 Python
使用html5 canvas绘制圆环动效
2019/06/03 HTML / CSS
Falconeri美国官网:由羊绒和羊毛制成的针织服装
2018/04/08 全球购物
美体小铺波兰官方网站:The Body Shop波兰
2019/09/03 全球购物
关于VPN
2012/06/10 面试题
机电工程专业应届生求职信
2013/10/03 职场文书
高三语文教学反思
2014/01/15 职场文书
入党转正介绍人意见
2015/06/03 职场文书
Vue Element UI自定义描述列表组件
2021/05/18 Vue.js
CSS实现单选折叠菜单功能
2021/11/01 HTML / CSS
MySQL Innodb索引机制详细介绍
2021/11/23 MySQL