#1024程序员节|征文#
1、计算属性 computed
在 Vue.js 中,计算属性(computed properties)是一种特殊的响应式属性,它们根据依赖的响应式数据自动更新。计算属性非常适合用于当你需要根据现有数据派生出一些状态时。
(1)、基本用法
计算属性是通过在组件的 computed
选项中定义的。它们使用一个函数来计算值,这个函数的返回值就是计算属性的值。这个函数会接收组件实例作为上下文(即 this
),并可以访问组件的响应式数据。
import { computed, ref } from 'vue';export default {setup() {const count = ref(0);// 定义一个计算属性const doubledCount = computed(() => count.value * 2);return {count,doubledCount};}
};
在上面的示例中,doubledCount
是一个计算属性,它依赖于 count
。每当 count
的值变化时,doubledCount
会自动更新。
(2)、缓存性
计算属性具有缓存性,这意味着只有当依赖的响应式数据变化时,计算属性才会重新计算。这可以提高性能,特别是当计算属性的计算成本较高时。
import { computed, ref } from 'vue';export default {setup() {const count = ref(0);// 定义一个计算属性const expensiveComputed = computed(() => {console.log('Computing expensive value...');return count.value * 2;});return {count,expensiveComputed};}
};
在这个例子中,每次 count
更新时,控制台都会打印 “Computing expensive value…”,但由于计算属性的缓存性,这个日志只会在 count
实际变化时出现。
(3)、不是所有场景都适合计算属性
虽然计算属性很强大,但它们并不适用于所有场景。如果某个值的计算不依赖于响应式数据,或者需要在每次访问时都进行计算(例如,随机数生成),则不应该使用计算属性。
(4)、计算属性 vs 方法
计算属性与方法(methods)不同。方法在每次调用时都会执行函数,而计算属性则具有缓存性,只有当依赖的数据变化时才会重新计算。因此,如果你需要在每次访问时都执行计算,应该使用方法而不是计算属性。
import { ref } from 'vue';export default {setup() {const now = ref(Date.now());// 使用方法而不是计算属性const timeString = () => now.value.toLocaleTimeString();return {now,timeString};}
};
2、计算属性应用场景
计算属性(computed properties)在 Vue.js 中适用于多种场景,尤其是当你需要根据组件中现有的响应式数据来声明派生数据时。以下是一些计算属性的应用场景示例:
(1)、根据多个数据属性计算新的数据
假设你有一个组件,它维护了用户的购物车信息,包括商品的原价和折扣后的价格。你可能需要计算商品的最终价格。
<template><div><p>原价: {{ product.originalPrice }}</p><p>折扣价: {{ product.discountedPrice }}</p><p>最终价格: {{ finalPrice }}</p></div>
</template><script>
import { computed, reactive } from 'vue';export default {setup() {const product = reactive({originalPrice: 100,discountedPrice: 80});const finalPrice = computed(() => product.originalPrice * product.discountedPrice / 100);return {product,finalPrice};}
};
</script>
(2)、根据用户输入动态计算结果
在表单中,你可能需要根据用户输入的数据动态计算一些结果,如总价、折扣、税费等。
<template><div><input v-model="quantity" type="number" placeholder="数量"><input v-model="pricePerItem" type="number" placeholder="单价"><p>总价: {{ totalPrice }}</p></div>
</template><script>
import { computed, ref } from 'vue';export default {setup() {const quantity = ref(1);const pricePerItem = ref(0);const totalPrice = computed(() => quantity.value * pricePerItem.value);return {quantity,pricePerItem,totalPrice};}
};
</script>
(3)、根据数组内容计算新的数组
如果你有一个商品列表,你可能需要根据这个列表计算出一些新的列表,比如筛选后的列表或排序后的列表。
<template><div><ul><li v-for="item in filteredProducts" :key="item.id">{{ item.name }} - {{ item.price }}</li></ul></div>
</template><script>
import { computed, ref } from 'vue';export default {setup() {const products = ref([{ id: 1, name: 'Product A', price: 100 },{ id: 2, name: 'Product B', price: 200 },// more products]);const filteredProducts = computed(() =>products.value.filter(product => product.price > 100));return {filteredProducts};}
};
</script>
(4)、根据对象数据计算新的值
如果你有一个对象,包含用户的基本信息,你可能需要根据这些信息计算出一些新的值,比如年龄、全名等。
<template><div><p>用户信息:</p><p>名字: {{ user.name }}</p><p>姓氏: {{ user.surname }}</p><p>全名: {{ fullName }}</p></div>
</template><script>
import { computed, reactive } from 'vue';export default {setup() {const user = reactive({name: 'John',surname: 'Doe',birthday: '1990-01-01'});const fullName = computed(() => `${user.name} ${user.surname}`);return {user,fullName};}
};
</script>
(5)、性能优化
计算属性可以用于性能优化,避免在模板或方法中重复执行相同的计算逻辑。
<template><div><p>列表长度: {{ listLength }}</p></div>
</template><script>
import { computed, ref } from 'vue';export default {setup() {const items = ref([/* ... */]);const listLength = computed(() => items.value.length);return {listLength};}
};
</script>
在这些示例中,计算属性允许你将复杂的逻辑放在 JavaScript 中处理,同时保持模板的简洁和声明性。这使得你的组件更容易维护和理解。
3、计算属性最佳实践
以下是一些使用计算属性的最佳实践:
(1)、 保持简洁
计算属性应该只包含必要的逻辑,用于根据响应式数据派生新的值。避免在计算属性中执行副作用操作,如 API 调用或更改其他响应式状态。
(2)、 避免复杂逻辑
如果计算属性的逻辑变得过于复杂,考虑将其拆分为多个更小的计算属性或方法。
(3)、 使用缓存
利用计算属性的缓存特性,避免不必要的计算。只有当依赖的数据变化时,计算属性才会重新计算。这意味着在模板或其他地方多次使用计算属性时,它只会计算一次。
(4)、 避免直接修改计算属性
计算属性是只读的,不应该直接修改。如果你需要修改计算属性的值,应该修改它的依赖数据。
(5)、选择合适的名称
给计算属性取一个清晰和描述性的名称,这样其他开发者可以很容易地理解这个计算属性的用途。
(6)、避免在计算属性中进行异步操作
计算属性应该基于同步操作,因为它们是响应式系统的同步部分。异步操作应该在方法或生命周期钩子中处理。
(7)、使用计算属性而不是观察者
如果你发现自己需要在多个地方执行相同的计算逻辑,考虑使用计算属性而不是使用 watch
或 `watchEffec
(8)、避免在模板中使用复杂表达式
对于模板中需要的复杂表达式,应该使用计算属性而不是在模板中直接编写复杂表达式。
(9)、组合使用计算属性和侦听器
在某些情况下,你可能需要在计算属性变化时执行额外的逻辑。这时,可以结合使用计算属性和 watch
侦听器。
(10)、考虑使用 watchEffect
如果你需要在多个响应式数据变化时执行副作用,考虑使用 watchEffect
而不是计算属性。
示例:计算属性和 watch
组合使用
<template><div>{{ fullName }}</div>
</template><script>
import { computed, watch, ref } from 'vue';export default {setup() {const firstName = ref('');const lastName = ref('');const fullName = computed(() => `${firstName.value} ${lastName.value}`);watch(fullName, (newName, oldName) => {console.log(`Name changed from ${oldName} to ${newName}`);// 执行一些副作用,比如 API 调用});return {fullName,firstName,lastName};}
};
</script>
在上述示例中,fullName
是一个计算属性,它依赖于 firstName
和 lastName
。我们使用 watch
来侦听 fullName
的变化,并在变化时执行副作用。