浅谈Vue.js的计算属性computed

简介: 浅谈Vue.js的计算属性computed

什么是computed属性

computed 属性用于声明计算属性,这些属性的值是基于其他响应式属性计算而来的,当依赖的响应式属性发生变化时,计算属性会自动重新计算。

与Vue.js 2相比,Vue.js 3的 computed 属性语法稍有变化,不再使用对象字面量,而是使用类似函数的形式来声明计算属性。此外,Vue.js 3还引入了一个新的 ref 函数,用于创建响应式引用,可以在 setup 函数中使用。

下面是一个简单的Vue.js 3中的 computed 的例子:

<template>
  <div>
    <p>{{ message }}</p>
    <p>{{ reversedMessage }}</p>
  </div>
</template>

<script setup>
// 使用 <script setup> 语法糖

// 定义响应式数据
const message = ref('Hello, Vue.js 3!');

// 使用 computed
const reversedMessage = computed(() => {
  // 计算属性的值是基于响应式数据 message 计算的
  return message.value.split('').reverse().join('');
});
</script>

在上面的例子中,reversedMessage 是一个计算属性,它依赖于 message 反序输出message的内容

计算属性的作用

computed 属性的存在主要是为了解决一些常见的需求,使得数据的计算和派生更加方便、高效,并确保数据的响应性。

以下是一些使用计算属性的主要原因:

  1. 「依赖追踪:」 计算属性允许你声明式地描述数据的依赖关系。当依赖的数据发生变化时,计算属性会自动重新计算,而无需手动编写更新逻辑。这有助于避免手动追踪依赖关系,提高代码的可维护性。
  2. 「缓存:」 计算属性会缓存其结果,只有在依赖发生变化时才重新计算。这意味着如果多次访问同一个计算属性,只有在它的依赖发生变化时才会重新计算,从而提高性能。
  3. 「简化模板逻辑:」 在模板中,你可以直接使用计算属性,而不必在模板中编写复杂的逻辑或计算。这使得模板更加清晰和易读。
  4. 「组合逻辑:」 计算属性允许你将一些复杂的逻辑组合成一个属性,使代码更加模块化和可复用。

下面是一个简单的例子,展示了计算属性的用途:

<script setup>
import { ref } from 'vue'
const radius = ref(5)
const area = computed(() => {
  return Math.PI * radius.value * radius.value
})

const circumference = computed(() => {
  return 2 * Math.PI * radius.value;
})
</script>

在上面的例子中,areacircumference 都是计算属性,它们依赖于 radius。当 radius 发生变化时,这两个计算属性会自动更新,而无需手动干预。这样可以使代码更加清晰和易于维护。

computed VS methods

计算属性 (computed 属性) 和普通的函数在Vue.js中的使用有一些区别和优势。计算属性更适合用于模板中的声明式逻辑,特别是涉及到响应式数据的复杂计算。普通函数更适合那些不依赖响应式数据的逻辑或者不需要自动依赖追踪的场景。

区别:

  1. 「自动依赖追踪:」
  • 「计算属性:」 Vue.js 会自动追踪计算属性的依赖关系。只要计算属性中用到的响应式数据发生变化,计算属性就会重新计算。
  • 「普通函数:」 普通函数没有自动的依赖追踪。你需要手动管理函数中使用的依赖关系,可能需要使用watch来监听变化,或者在模板中使用函数时手动触发更新。
  1. 「缓存机制:」
  • 「计算属性:」 具有缓存机制,只有当依赖变化时才会重新计算。多次访问相同计算属性时,只会计算一次。
  • 「普通函数:」 没有内置的缓存机制,每次调用函数都会重新执行。

优势:

  1. 「简化模板逻辑:」
  • 「计算属性:」 用于在模板中声明式地处理复杂的逻辑,使模板更加清晰和简洁。
  • 「普通函数:」 在模板中使用普通函数可能导致模板变得复杂,尤其是当逻辑比较复杂时。
  1. 「性能优化:」
  • 「计算属性:」 具有缓存机制,可以避免不必要的重复计算,提高性能。
  • 「普通函数:」 没有缓存机制,每次调用都会重新执行,可能导致性能下降。
  1. 「代码组织:」
  • 「计算属性:」 用于将相关逻辑组织成属性,使代码更加模块化和可维护。
  • 「普通函数:」 在组件中直接定义函数,可能导致代码分散,难以维护。

可写setter

在Vue.js中,计算属性默认是只读的,也就是说你不能直接在模板中通过v-model或者类似的方式修改计算属性的值。计算属性是依赖于其他响应式数据的,它的值是由这些响应式数据计算而来的。

如果你需要在Vue实例中有一个既能够计算值,又能够被修改的属性,你可以使用 computedgetset 方法。这样你就可以通过 v-model 或者手动赋值的方式修改这个属性的值。

以下是一个例子:

<template>
  <div>
    <p>Radius: {{ radius }}</p>
    <p>Area: {{ area }}</p>

    <input v-model="radius" type="number" placeholder="Enter radius" />
  </div>
</template>

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

const radius = ref(5);

const area = computed({
  get: () => Math.PI * radius.value * radius.value,
  set: (newValue) => {
    // 当修改 area 时,更新 radius
    radius.value = Math.sqrt(newValue / Math.PI);
  }
});
</script>

在上面的例子中,area 是一个计算属性,通过 get 方法计算值,通过 set 方法监听对 area 的修改,然后反向计算出对应的 radius。这样你就可以在模板中使用 v-model="area" 来修改 area 的值。

使用getter注意事项

计算属性的 getter 主要用于计算和返回一个派生值,应当保持简单、同步,不应该执行复杂的逻辑或副作用。如果有复杂逻辑或异步操作,应当考虑使用其他适当的方式。当使用计算属性的 getter 时,有一些需要注意的事项:

  1. 「只返回值:」 计算属性的 getter 应当只返回一个值,而不是执行一些可能产生副作用的操作。计算属性的目的是计算一个值,而不是用于执行命令式的操作。如果你需要执行一些副作用,应该考虑使用生命周期钩子函数或 watch
  2. 「不要使用箭头函数:」getter 中,尽量不要使用箭头函数。因为箭头函数没有自己的 this 上下文,而在计算属性中,this 指向的是当前 Vue 实例,而非调用它的对象。使用普通函数确保正确的 this 上下文。当然在vue3 setup语法糖内是可以使用的箭头函数的。
  3. 「避免异步操作:」 计算属性的 getter 应当是同步的,不要在 getter 中执行异步操作。如果需要异步操作,可以考虑使用 watch 或者其他适当的生命周期钩子。
  4. 「避免修改计算属性依赖的响应式数据:」 计算属性依赖于响应式数据,但在 getter 中应当避免修改这些响应式数据,因为这样会导致无限循环更新。
computed: {
  myComputedProperty: function() {
    // 避免在 getter 中修改依赖的响应式数据
    // 这样会导致无限循环更新
    this.someValue = this.someValue + 1;
    return this.someValue;
  }
}

小结

  • 计算属性值会基于其响应式依赖被缓存。一个计算属性仅会在其响应式依赖更新时才重新计算。
  • 可以使用setter修改计算属性的值
  • 不要在 getter 中做异步请求或者更改 DOM
目录
相关文章
|
2月前
|
JavaScript 前端开发 程序员
前端原生Js批量修改页面元素属性的2个方法
原生 Js 的 getElementsByClassName 和 querySelectorAll 都能获取批量的页面元素,但是它们之间有些细微的差别,稍不注意,就很容易弄错!
|
2月前
|
JavaScript 前端开发
|
2月前
|
监控 JavaScript 前端开发
确定使用 `defer` 属性还是 `async` 属性来异步加载 JavaScript
【10月更文挑战第24天】选择使用 `defer` 属性还是 `async` 属性来异步加载 JavaScript 是一个需要综合考虑多个因素的决策。需要根据脚本之间的依赖关系、页面加载性能要求、脚本的功能和重要性等因素来进行权衡。在实际应用中,需要通过测试和验证来确定最适合的加载方式,以提供更好的用户体验和页面性能。
|
2月前
|
存储 缓存 JavaScript
如何在大型 Vue 应用中有效地管理计算属性和侦听器
在大型 Vue 应用中,合理管理计算属性和侦听器是优化性能和维护性的关键。本文介绍了如何通过模块化、状态管理和避免冗余计算等方法,有效提升应用的响应性和可维护性。
|
2月前
|
监控 JavaScript 前端开发
使用 `defer` 属性异步加载 JavaScript
【10月更文挑战第24天】使用 `defer` 属性异步加载 JavaScript 是一种有效的提高页面性能和用户体验的方法。通过合理设置 `defer` 属性,可以在不影响页面渲染的情况下异步加载脚本,并确保脚本的执行顺序。在实际应用中,需要根据具体情况选择合适的加载方式,并注意处理可能出现的问题,以确保页面能够正常加载和执行。
|
3月前
|
监控 JavaScript 开发者
在 Vue 中,子组件为何不可以修改父组件传递的 Prop,如果修改了,Vue 是如何监控到属性的修改并给出警告的
在 Vue 中,子组件不能直接修改父组件传递的 Prop,以确保数据流的单向性和可预测性。如果子组件尝试修改 Prop,Vue 会通过响应式系统检测到这一变化,并在控制台发出警告,提示开发者避免这种操作。
|
3月前
|
JavaScript 前端开发 开发者
VUE学习一:初识VUE、指令、动态绑定、计算属性
这篇文章主要介绍了Vue.js的基础知识,包括初识Vue、指令如v-for、v-on的使用、动态属性绑定(v-bind)、计算属性等概念与实践示例。
72 1
|
3月前
|
缓存 JavaScript Serverless
vue中computed计算属性、watch侦听器、methods方法的区别以及用法
vue中computed计算属性、watch侦听器、methods方法的区别以及用法
214 0
|
3月前
|
存储 JavaScript 前端开发
js中map属性
js中map属性
30 0
|
20天前
|
JavaScript
vue使用iconfont图标
vue使用iconfont图标
109 1