前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >Vue的computed和watch的细节全面分析

Vue的computed和watch的细节全面分析

作者头像
火狼1
发布于 2019-04-17 05:38:18
发布于 2019-04-17 05:38:18
1.9K00
代码可运行
举报
运行总次数:0
代码可运行

1.computed

1.1 定义

是一个计算属性,类似于过滤器,对绑定到view的数据进行处理

1.2 get用法

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  data: {
    firstName: 'Foo',
    lastName: 'Bar'
  },
  computed: {
    fullName: function () {
      return this.firstName + ' ' + this.lastName
    }
  }

fullName不可在data里面定义, 如果定义会报如下图片的错误,因为对应的computed作为计算属性定义fullName并返回对应的结果给这个变量,变量不可被重复定义和赋值

1.3 get和set用法

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
data: {
    firstName: 'Foo',
    lastName: 'Bar'
  },
  computed: {
  fullName:{
   get(){//回调函数 当需要读取当前属性值是执行,根据相关数据计算并返回当前属性的值
      return this.firstName + ' ' + this.lastName
    },
   set(val){//监视当前属性值的变化,当属性值发生变化时执行,更新相关的属性数据
       //val就是fullName的最新属性值
       console.log(val)
        const names = val.split(' ');
        console.log(names)
        this.firstName = names[0];
        this.lastName = names[1];
   }
   }
  }

2. watch

2.1 定义

watch是一个观察的动作

2.2 示例

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  data: {
    firstName: 'Foo',
    lastName: 'Bar',
    fullName: 'Foo Bar'
  },
  watch: {
     firstName: function (val) {
     this.fullName = val + ' ' + this.lastName
  },
  lastName: function (val) {
     this.fullName = this.firstName + ' ' + val
   }
   }

上面是监听firstName和lastName的变化,但是仅限简单数据类型

2.2 监听简单数据类型

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
data(){
      return{
        'first':2
      }
    },
    watch:{
      first(){
        console.log(this.first)
      }
    },

2.3 监听复杂数据类型

1.监听复杂数据类型需用深度监听

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
data(){
      return{
        'first':{
          second:0
        }
      }
    },
    watch:{
      secondChange:{
        handler(oldVal,newVal){
          console.log(oldVal)
          console.log(newVal)
        },
        deep:true
      }
    },

2.console.log打印的结果,发现oldVal和newVal值是一样的,所以深度监听虽然可以监听到对象的变化,但是无法监听到具体对象里面那个属性的变化

3.oldVal和newVal值一样的原因是它们索引同一个对象/数组。Vue 不会保留修改之前值的副本 vm.$watch的深度监听

4.深度监听对应的函数名必须为handler,否则无效果,因为watcher里面对应的是对handler的调用

2.4 监听对象单个属性

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
方法一:可以直接对用对象.属性的方法拿到属性
data(){
          return{
            'first':{
              second:0
            }
          }
        },
        watch:{
          first.second:function(newVal,oldVal){
            console.log(newVal,oldVal);
          }
        },

方法二:watch如果想要监听对象的单个属性的变化,必须用computed作为中间件转化,因为computed可以取到对应的属性值

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
data(){
      return{
        'first':{
          second:0
        }
      }
    },
    computed:{
      secondChange(){
        return this.first.second
      }
    },
    watch:{
      secondChange(){
        console.log('second属性值变化了')
      }
    },

3 computed和watch的区别

3.1 computed特性

1.是计算值, 2.应用:就是简化tempalte里面{{}}计算和处理props或$emit的传值 3.具有缓存性,页面重新渲染值不变化,计算属性会立即返回之前的计算结果,而不必再次执行函数

3.2 watch特性

1.是观察的动作, 2.应用:监听props,$emit或本组件的值执行异步操作 3.无缓存性,页面重新渲染时值不变化也会执行

3 props传值

3.1 常见错误1

传入的值想作为局部变量来使用,直接使用会

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
props:['listShop'],
    data(){
      return{}
    },
    created(){
      this.listShop=30
}

报错

这个错误是说的避免直接修改父组件传入的值,因为会改变父组件的值,贴上官网介绍

3.2 解决方案1

简单数据类型解决方案: 所以可以在data中重新定义一个变量,改变指向,但是也只是针对简单数据类型,因为复杂数据类型栈存贮的是指针,

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
props:['listShop'],
    data(){
      return{
        listShopChild:this.listShop
      }
    },
    created(){
      this.listShopChild=30
    }

这样就可以愉快的更改传入的简单数据类型的数据啦!不会有任何报错,也不会影响父组件!

3.4 存在的问题

复杂数据类型在栈中存贮的是指针,所以赋值给新的变量也会改变原始的变量值.那么应该咋整呢? 1.可以手动深度克隆一个复杂的数据出来,循环或者递归都行

数组深度克隆:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var x = [1,2,3];
var y = [];
for (var i = 0; i < x.length; i++) {
    y[i]=x[i];
}
console.log(y);  //[1,2,3]
y.push(4);
console.log(y);  //[1,2,3,4]
console.log(x);  //[1,2,3]

对象深度克隆:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var x = {a:1,b:2};
var y = {};
for(var i in x){
    y[i] = x[i];
}
console.log(y);  //Object {a: 1, b: 2}
y.c = 3;
console.log(y);  //Object {a: 1, b: 2, c: 3}
console.log(x);  //Object {a: 1, b: 2}

函数深度克隆

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var x = function(){console.log(1);};
var y = x;
y = function(){console.log(2);};
x();  //1
y();  //2

为什么函数可以直接赋值克隆? 由于函数对象克隆之后的对象会单独复制一次并存储实际数据,因此并不会影响克隆之前的对象。所以采用简单的复制“=”即可完成克隆。

2.Object.assign 只会对只是一级属性复制,比浅拷贝多深拷贝了一层而已,所以还是无法达到深度克隆的目的. 详请请戳

3.强大的JSON.stringify和JSON.parse

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const obj1 = JSON.parse(JSON.stringify(obj));

这是ES5新出来的API,先将对象转化为字符串,就是简单数据类型赋值,再用JSON.parse转化

3.5 解决方案2

直接用computed改变

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
computed:{
  listShopChild(){
    return this.listShop
   }
}

3.5 存在的问题

注意:此时用computed时,如果是数组this.$set(arr,1,true)对应的值耶不更新, 这个很坑,这个bug我找个很久 如果传入的值只是在data定义,并未在methods或生命周期钩子更改,直接改变也会报错 所以还是可以先用局部变量接收,再修改,这个坑比较多

4 应用

4.1 应用1

监听本组件计算和监听

4.2 应用2

计算或监听父传子的props值

4.3 应用3

分为简单数据类型和复杂数据类型监听,监听方法如上watch的使用

4.4 应用4

监听vuex的state或者getters值的变化

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
computed:{
    stateDemo(){
        return this.$store.state.demoState;
    }
}
watch:{
    stateDemo(){
        console.log('vuex变化啦')
    }
}

5. computed和watch的原理分析

很开心小伙伴们能看到这里,接下来给大家简单罗列下他们的原理!

5.1 computed的原理

深入理解 Vue Computed 计算属性

5.2 watch的原理

分为三个过程:实例化Vue、调用$watch方法、属性变化,触发回调 Vue的数据依赖实现原理简析 vue中$watch源码阅读笔记

6 简单实现computed和watch

公共类

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
function defineReactive(data, key, val, fn) {
      let subs = [] // 新增
      Object.defineProperty(data, key, {
        configurable: true,
        enumerable: true,
        get: function() {
          // 新增
       if (data.$target) {
        subs.push(data.$target)
      }
      return val
     },
     set: function(newVal) {
      if (newVal === val) return
      fn && fn(newVal)
      // 新增
      if (subs.length) {
        // 用 setTimeout 因为此时 this.data 还没更新
        setTimeout(() => {
          subs.forEach(sub => sub())
        }, 0)
      }
      val = newVal
    },
   })
 }

6.1 computed实现

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
function computed(ctx, obj) {
  let keys = Object.keys(obj)
  let dataKeys = Object.keys(ctx.data)
  dataKeys.forEach(dataKey => {
    defineReactive(ctx.data, dataKey, ctx.data[dataKey])
  })
  let firstComputedObj = keys.reduce((prev, next) => {
    ctx.data.$target = function() {
      ctx.setData({ [next]: obj[next].call(ctx) })
    }
    prev[next] = obj[next].call(ctx)
    ctx.data.$target = null
    return prev
  }, {})
  ctx.setData(firstComputedObj)
}

6.1 watch实现

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
function watch(ctx, obj) {
  Object.keys(obj).forEach(key => {
    defineReactive(ctx.data, key, ctx.data[key], function(value) {
      obj[key].call(ctx, value)
    })
  })
}

https://segmentfault.com/a/11... 大家如果发现有什么错误,欢迎指正,共同交流。如果觉得篇文章真的对你有点作用。 谢谢亲们能看完!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Vue中computed和watch比较
Vue在监听数据的时候我们都会用computed和watch,但是很少人真正的去比较两个的区别,一般都是能实现就行。本人也是,习惯了watch,之后就几乎没用过computed。所以用官网的例子理解一下这两者的区别。
wade
2020/04/24
7190
十六.Vue监控数据
<input type="text" v-model="firstname" @keyup="getFullname">
Yuyy
2022/06/28
3140
Vue核心知识:computed、methods和watch的区别
当我们改变 number 时,整个应用会重新渲染,vue 会被数据重新渲染到 dom 中。
xyzzz
2020/09/26
6K0
浅析 Vue 的 `watch` 函数
在 Vue.js 中,响应式系统是其核心特性之一,通过它可以轻松地跟踪数据变化并自动更新视图。而 watch 函数则是 Vue 提供的一种用于监听和响应数据变化的高级方法。在这篇博客中,我们将深入探讨 watch 函数的使用方法、应用场景以及一些常见的陷阱。
繁依Fanyi
2024/07/06
2540
vue --- watch 高级用法
上面的代码的效果是,当我们输入firstName后,wacth监听每次修改变化的新值,然后计算输出fullName。
小蔚
2019/09/11
1.2K0
vue学习笔记4
考虑一个问题:想要实现 名 和 姓 两个文本框的内容改变,则全名的文本框中的值也跟着改变;(用以前的知识如何实现???)
用户6362579
2019/09/29
5940
Vue.js中watch的高级用法
复制代码上面的代码的效果是,当我们输入firstName后,wacth监听每次修改变化的新值,然后计算输出fullName。
Tiffany_c4df
2019/09/04
2.1K0
51. Vue名称案例-使用watch监听数据变化
上一章节,我才用了监听keyup事件的方式,实现了一个名称拼接的案例。那么其中Vue框架提供一个watch组件,可以用来监听数据的变化,然后再执行相关的业务方法。
Devops海洋的渔夫
2020/08/11
1.2K0
51. Vue名称案例-使用watch监听数据变化
53. Vue名称案例-使用computed计算属性
前面在写名称拼接案例的时候使用了keyup监听文本框,还使用了watch来监听数据变化,那么本篇章来使用computed计算属性来监听实现。
Devops海洋的渔夫
2022/01/17
5770
53. Vue名称案例-使用computed计算属性
Vue常用特性-计算属性computed 和 侦听器 watch
1.4.computed比较适合对多个变量或者对象进行处理后返回一个结果值,也就是数多个变量中的某一个值发生了变化则我们监控的这个值也就会发生变化
梨涡浅笑
2020/12/02
5650
Vue常用特性-计算属性computed 和 侦听器  watch
vue教程:Vue.js中 watch 的高级用法
当我们修改firstName后,wacth监听每次修改变化的新值,然后计算输出fullName,但是这个代码有一个缺陷,就是默认初始化时候,fullName是没有值的,这里 watch 的一个特点是,最初绑定的时候是不会执行的。
用户4831957
2019/07/31
1.8K0
全面分析 Vue 的 computed 和 watch 的区别
computed 用来监控自己定义的变量,该变量在 data 内没有声明,直接在 computed 里面定义,页面上可直接使用。
呆呆
2021/12/01
2810
10天从入门到精通Vue(四)Vue中的路由指南
考虑一个问题:想要实现 名 和 姓 两个文本框的内容改变,则全名的文本框中的值也跟着改变;
共饮一杯无
2022/11/28
5070
10天从入门到精通Vue(四)Vue中的路由指南
vue学习笔记3
什么是组件: 组件的出现,就是为了拆分Vue实例的代码量的,能够让我们以不同的组件,来划分不同的功能模块,将来我们需要什么样的功能,就可以去调用对应的组件即可; 组件化和模块化的不同:
用户6362579
2019/09/29
7550
(2)打鸡儿教你Vue.js
钩子函数 - beforeCreate() - 实例初始化之后 钩子函数 - created() - 发送请求获取数据 钩子函数 - beforeMounted() - 在挂载开始之前被调用 钩子函数 - mounted() - vue实例已经挂载到页面中 钩子函数 - beforeUpdated() 钩子函数 - updated() 钩子函数 - beforeDestroy() 钩子函数 - destroyed()
达达前端
2019/07/03
5780
(2)打鸡儿教你Vue.js
Vue 侦听器 watch 扩展之立即触发回调、深度监听和注销
原来我们 watch 中默认写的就是这个 handler,Vue 会去处理这个逻辑,最终编译出来其实就是这个 handler
Leophen
2020/08/02
2.7K0
常见经典vue面试题(面试必问)
如果不使用 key,Vue 会使用一种最大限度减少动态元素并且尽可能的尝试就地修改/复用相同类型元素的算法。key 是为 Vue 中 vnode 的唯一标记,通过这个 key,我们的 diff 操作可以更准确、更快速
bb_xiaxia1998
2022/12/14
9140
深入理解Vue中的计算属性与监听属性
在Vue.js这个构建用户界面的渐进式框架中,计算属性(Computed Properties)和监听属性(Watch Properties)是处理数据和响应式更新的两个核心特性。它们为开发者提供了不同的方式来操作和响应数据的变化,正确理解和合理运用这两个概念对于开发高效、可维护的Vue应用具有不可忽视的重要性。
Front_Yue
2025/01/12
1020
深入理解Vue中的计算属性与监听属性
【Vue】谈Vue的依赖追踪系统 ——搞懂methods watch和compute的区别和联系
从作用机制和性质上看待methods,watch和computed的关系 图片标题[原创]:《他三个是啥子关系呢?》 首先要说,methods,watch和computed都是以函数为基础的,但各自却
啦啦啦321
2018/01/03
1.2K0
【Vue】谈Vue的依赖追踪系统 ——搞懂methods watch和compute的区别和联系
Vue学习笔记②
计算属性将被混入到 Vue 实例中。所有 getter 和 setter 的 this 上下文自动地绑定为 Vue 实例。
ymktchic
2022/01/18
6830
Vue学习笔记②
相关推荐
Vue中computed和watch比较
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文