官方解释:一个对象,系统定制开发键是需要观察的表达式,系统定制开发值是对应回调函数。系统定制开发值也可以是方法名,系统定制开发或者包含选项的对象。Vue 系统定制开发实例将会在实例化时调用 watch,遍历 watch 系统定制开发对象的每一个 属性。
示例:
- <template>
- <el-card class="box-card"><el-input v-model="name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- name: '123'
- };
- }
- };
- </script>
-
- <style></style>
第一种:常规用法
(1)把要监听的name值看作方法名,来进行监听。【第一种写法】
- <template>
- <el-card class="box-card"><el-input v-model="name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- name: '123'
- };
- },
- watch: {
- name(newVal, oldVal) {
- console.log('newVal', newVal);// 1234
- console.log('oldVal', oldVal);// 123
- }
- }
- };
- </script>
-
- <style></style>
(2)把要监听的name值看作对象,利用hanler方法来进行监听。【第二种写法】
- <template>
- <el-card class="box-card"><el-input v-model="name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- name: '123'
- };
- },
- watch: {
- name:{
- handler(newVal,oldVal){
- console.log('newVal',newVal); // 1234
- console.log('oldVal',oldVal); // 123
- }
- }
- }
- };
- </script>
-
- <style></style>
以上两种写法是器的普通用法,这种用法有一个特点,就是当值第一次绑定的时候,不会执行监听函数,只有当值发生改变时才会执行。如果我们需要在最初绑定值的时侯,也执行监听函数,则就需要用到immediate属性。
下面,我们就往高级一点的用法上讲。
第二种:高级用法
比如,当父组件向子组件动态传值时,子组件props首次获取到父组件传来的默认值时,也需要执行函数,此时就需要将属性设置为true,结合handler方法使用。
当设置immediate属性为true时,无论值是否发生改变,时刻都会监听;
当设置immediate属性为false时,常规用法,只有值发生改变才会监听。
- <template>
- <el-card class="box-card"><el-input v-model="name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- name: '123'
- };
- },
- watch: {
- name: {
- handler(newVal, oldVal) {
- console.log('newVal', newVal);
- console.log('oldVal', oldVal);
- },
- immediate: true
- }
- }
- };
- </script>
-
- <style></style>
立即执行:
值改变时:
第三种:超高级用法(deep 深度监听)
(1)监听普通变量的变化可以使用以上两种方法,但是要监听变量值是某对象的时候,则不起作用。
例如,我们监听form对象内部属性的变化,是监听不到的。
- <template>
- <el-card class="box-card"><el-input v-model="form.name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- form: {
- name: '123'
- }
- };
- },
- watch: {
- form: {
- handler(newVal, oldVal) {
- console.log('newVal', newVal);
- console.log('oldVal', oldVal);
- }
- }
- }
- };
- </script>
-
- <style></style>
则,从结果来看,我们没有看到任何的输出打印,所以普通的watch方法无法监听到对象内部属性的变化。
那么,我们该怎么办才能监听到对象内部属性的变化呢?
watch方法提供了一个deep属性(深度监听),该属性可以监听到对象内部属性的改变。
- <template>
- <el-card class="box-card"><el-input v-model="form.name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- form: {
- name: '123'
- }
- };
- },
- watch: {
- form: {
- handler(newVal, oldVal) {
- console.log('newVal', newVal);
- console.log('oldVal', oldVal);
- },
- deep: true
- }
- }
- };
- </script>
-
- <style></style>
设置deep: true 则可以监听到form的变化,如果form有较多属性的话,此时会给form的所有属性都会加上这个监听器,每个属性值的变化都会执行handler。
当deep属性值为true时,就可以监听到对象属性内部的改变;
当deep属性值为false时,则监听不到。
(2)如果只需要监听对象中的某一个属性值时,我们可以使用:字符串的形式监听对象属性,
这个监听过程,不需要使用deep去深度监听,就可以监听对象中某个属性的变化。
- <template>
- <el-card class="box-card"><el-input v-model="form.name" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- form: {
- name: '123'
- }
- };
- },
- watch: {
- 'form.name': {
- handler(newVal, oldVal) {
- console.log('newVal', newVal);
- console.log('oldVal', oldVal);
- }
- }
- }
- };
- </script>
-
- <style></style>
第四种:扩展(监听数组)
(1)(一维、多维)数组的变化不需要深度监听
- <template>
- <el-card class="box-card"><el-input v-model="name" @input="inputFn" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- name: '123',
- arr1: [1, 2, 3],
- arr2: [1, 2, 3, [4, 5]]
- };
- },
- watch: {
- arr1(newVal, oldVal) {
- console.log('newVal1', newVal);
- console.log('oldVal1', oldVal);
- },
- arr2(newVal, oldVal) {
- console.log('newVal2', newVal);
- console.log('oldVal2', oldVal);
- }
- },
- methods: {
- inputFn(e) {
- this.arr1.push(e);
- this.arr2.push(e);
- }
- }
- };
- </script>
-
- <style></style>
(2)数组对象中对象属性变化监测需要使用deep:true深度监听,多少层内产生变化都可以监测到。
- <template>
- <el-card class="box-card"><el-input v-model="name" @input="inputFn" style="width: 30%;"></el-input></el-card>
- </template>
-
- <script>
- export default {
- data() {
- return {
- name: '123',
- arr1: [
- {
- id: 1,
- sex: 11
- }
- ],
- arr2: [
- {
- id: 2,
- sex: 22,
- list: [
- {
- id: 3,
- sex: 33
- }
- ]
- }
- ]
- };
- },
- watch: {
- arr1: {
- handler(newVal, oldVal) {
- console.log('newVal1', newVal);
- console.log('oldVal1', oldVal);
- },
- deep: true
- },
- arr2: {
- handler(newVal, oldVal) {
- console.log('newVal2', newVal);
- console.log('oldVal2', oldVal);
- },
- deep: true
- }
- },
- methods: {
- inputFn(e) {
- this.arr1[0].sex = e;
- this.arr2[0].list[0].sex = e;
- }
- }
- };
- </script>
-
- <style></style>