computed与watch到底有什么区别?

咱也不去深刨什么源码。

从实际使用来讲就一句话:你computed能做的,我watch能做,你computed做不了的,我watch也能做!

只是从代码的优雅度来讲,因为watch是根据响应式副作用触发的回调,在这个回调函数里面,可以做任何前提为数据发生变化所想执行的操作,并且能拿到修改之前的值。

computed是依赖于一个响应式数据,当这个数据发生改变,执行方法根据这个数据衍生出一个新数据。并且计算属性值会基于其响应式依赖被缓存。如果依赖的数据没有发生变化,就可以直接返回缓存的值,避免了重复计算的开销。 这个也是它还能存在的原因。不然直接watch一把梭了。

下面是几个场景分别看看watchcomputed是如何实现的。

场景一:需要监听一个响应式变量,一旦变量的值发生变化就执行里面代码。

js 复制代码
<template>
  <div class="difference">
    <h2>我是数字 {{ count }}</h2>
    <h2>我是watch {{ watchDoubleCount }}</h2>
    <h2>我是computed {{ computedDoubleCount }}</h2>
    <el-button type="primary" @click="countClick">按钮</el-button>
  </div>
</template>

<script setup>
import { ref, watch, computed } from 'vue'
const count = ref(1)
const watchDoubleCount = ref(null)

function countClick() {
  count.value++
}

// watch示例
// 每次count变化时,watch中的回调函数都会被调用。
watch(
  count,
  (newVal, oldVal) => {
    //数据发生变化所想执行的操作
    console.log('------watch------')
    watchDoubleCount.value = newVal * 2
    console.log('新值:' + newVal, '旧值:' + oldVal)
  },
)
// computed示例
// computed会自动跟踪count的变化,并在count变化时重新计算computedDoubleCount的值。
const computedDoubleCount = computed(() => {
  console.log('------computed------')
  console.log('新值:' + count.value)
  console.log('旧值:无从获取')
  //根据这个数据衍生出一个新数据。
  return count.value * 2
})
</script>

还没执行任何操作时(还没点击按钮),因为watch是根据回调函数执行还没有被触发,而计算属性computed会在页面初始化时立即执行一次,是为了确保初始值是正确的,并将其缓存起来。当然给watch传入第三个参数{immediate: true}也是可以立即执行一次的。


点击按钮对数字进行了更改,可以看到里面方法都有被执行,效果是一样的。

但是如果是下面这种情况computed就无能为力了

场景二:需要监听一个对象,一旦对象的某条属性发生变化就执行里面代码。注意是某一条。

js 复制代码
<template>
  <div class="difference">
    <h2>我是数字 {{ count.salary }}</h2>
    <el-button type="primary" @click="countClick">按钮</el-button>
  </div>
</template>

<script setup>
import { ref, watch, computed } from 'vue'

const count = ref({
  name: '张三',
  age: 25,
  gender: '男',
  occupation: '工程师',
  salary: 1000,
  isMarried: false
})

function countClick() {
  count.value.salary++
}

//watch示例
watch(
  count,
  (newVal, oldVal) => {
    console.log('------watch------')
    // watchDoubleCount.value = newVal.salary * 2
    // console.log('新值:' + newVal.salary, '旧值:' + newVal.salary)
  },
  { deep: true }
)

// computed示例
const computedDoubleCount = computed(() => {
  console.log('------computed------')
  console.log(count.value)
  // console.log('新值:' + count.value.salary)
  // console.log('旧值:无从获取')
})

因为如果给computed确切的去指出哪一条属性的话,那么它会自动添加依赖。这并不是我想要的某一条情况,另外这种情况的话需要给 watch增加第三个参数{ deep: true }深度侦听。

看效果

能看出来只有watch实现了,里面的操作被多次执行。

因此,在实际使用中,如果只是简单的数据监听和计算,推荐使用computed,因为它会带来一定的性能提升。而对于一些需要执行副作用或者比较复杂的逻辑操作的情况,可以选择使用watch

相关推荐
m0_5287238134 分钟前
HTML中,title和h1标签的区别是什么?
前端·html
Dark_programmer35 分钟前
html - - - - - modal弹窗出现时,页面怎么能限制滚动
前端·html
GDAL40 分钟前
HTML Canvas clip 深入全面讲解
前端·javascript·canvas
禾苗种树41 分钟前
在 Vue 3 中使用 ECharts 制作多 Y 轴折线图时,若希望 **Y 轴颜色自动匹配折线颜色**且无需手动干预,可以通过以下步骤实现:
前端·vue.js·echarts
贵州数擎科技有限公司1 小时前
使用 Three.js 实现流光特效
前端·webgl
JustHappy1 小时前
「我们一起做组件库🌻」做个面包屑🥖,Vue的依赖注入实战💉(VersakitUI开发实录)
前端·javascript·github
拉不动的猪1 小时前
刷刷题16
前端·javascript·面试
哑巴语天雨2 小时前
前端面试-网络协议篇
websocket·网络协议·http·面试·https
01_2 小时前
力扣hot100——LRU缓存(面试高频考题)
leetcode·缓存·面试·lru
祈澈菇凉3 小时前
如何结合使用thread-loader和cache-loader以获得最佳效果?
前端