Vue.js 是一个构建用户界面的 JavaScript 框架,它的核心思想是数据驱动,即组件的状态由响应式数据来驱动,当数据发生变化时,视图会进行相应的更新。Vue.js 中有两个重要的概念:计算属性和侦听器。
计算属性是一个特殊的 Vue 功能,它允许你将一些复杂逻辑封装在一个函数中,然后在你的组件中使用它。它可以帮助你将复杂的逻辑从组件中分离出来,使代码变得更加可读、易于理解。
// 这是一个计算属性 computed: { reversedMessage() { return this.message.split('').reverse().join('') } }
侦听器也是 Vue 的一个重要特性,它允许你监听 Vue 对象上的数据变化,并执行相应的回调函数。侦听器可以帮助你在数据发生变化时执行特定的逻辑。
// 这是一个侦听器 watch: { message(newVal, oldVal) { console.log('message changed from ' + oldVal + ' to ' + newVal) } }
因此,我们可以看到:
模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。例如,有一个嵌套数组对象:
Vue.createApp({
data() {
return {
author: {
name: "John Doe",
books: [
"Vue 2 - Advanced Guide",
"Vue 3 - Basic Guide",
"Vue 4 - The Mystery"
]
}
}
}
})
我们想根据 author
是否已经有一些书来显示不同的消息
<div id="computed-basics">
<p>Has published books:</p>
<span>{{ author.books.length > 0 ? "Yes" : "No" }}</span>
</div>
此时,模板不再是简单的和声明性的。你必须先看一下它,然后才能意识到它执行的计算取决于 author.books
。如果要在模板中多次包含此计算,则问题会变得更糟。
所以,对于任何包含响应式数据的复杂逻辑,你都应该使用计算属性。
<div id="computed-basics">
<p>Has published books:</p>
<span>{{ publishedBooksMessage }}</span>
</div>
Vue.createApp({
data() {
return {
author: {
name: "John Doe",
books: [
"Vue 2 - Advanced Guide",
"Vue 3 - Basic Guide",
"Vue 4 - The Mystery"
]
}
}
},
computed: {
// 计算属性的 getter
publishedBooksMessage() {
// `this` points to the vm instance
return this.author.books.length > 0 ? "Yes" : "No"
}
}
}).mount("#computed-basics")
Result: 点击此处实现
这里声明了一个计算属性 publishedBooksMessage
。
尝试更改应用程序 data
中 books
数组的值,你将看到 publishedBooksMessage
如何相应地更改。
你可以像普通属性一样将数据绑定到模板中的计算属性。Vue 知道 vm.publishedBookMessage
依赖于 vm.author.books
,因此当 vm.author.books
发生改变时,所有依赖 vm.publishedBookMessage
绑定也会更新。而且最妙的是我们已经声明的方式创建了这个依赖关系:计算属性的 getter 函数没有副作用,这使得更易于测试和理解。
你可能已经注意到我们可以通过在表达式中调用方法来达到同样的效果:
<p>{{ calculateBooksMessage() }}</p>
// 在组件中
methods: {
calculateBooksMessage() {
return this.author.books.length > 0 ? "Yes" : "No"
}
}
我们可以将同一函数定义为一个方法而不是一个计算属性。两种方式的最终结果确实是完全相同的。然而,不同的是计算属性是基于它们的反应依赖关系缓存的。计算属性只在相关响应式依赖发生改变时它们才会重新求值。这就意味着只要 author.books
还没有发生改变,多次访问 publishedBookMessage
计算属性会立即返回之前的计算结果,而不必再次执行函数。
这也同样意味着下面的计算属性将不再更新,因为 Date.now () 不是响应式依赖:
computed: {
now() {
return Date.now()
}
}
相比之下,每当触发重新渲染时,调用方法将总会再次执行函数。
我们为什么需要缓存?假设我们有一个性能开销比较大的计算属性 list
,它需要遍历一个巨大的数组并做大量的计算。然后我们可能有其他的计算属性依赖于 list
。如果没有缓存,我们将不可避免的多次执行 list
的 getter!如果你不希望有缓存,请用 method
来替代。
计算属性默认只有 getter,不过在需要时你也可以提供一个 setter:
// ...
computed: {
fullName: {
// getter
get() {
return this.firstName + " " + this.lastName
},
// setter
set(newValue) {
const names = newValue.split(" ")
this.firstName = names[0]
this.lastName = names[names.length - 1]
}
}
}
// ...
现在再运行 vm.fullName = "John Doe"
时,setter 会被调用,vm.firstName
和 vm.lastName
也会相应地被更新。
虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的侦听器。这就是为什么 Vue 通过 watch
选项提供了一个更通用的方法,来响应数据的变化。当需要在数据变化时执行异步或开销较大的操作时,这个方式是最有用的。
例如:
<div id="watch-example">
<p>
Ask a yes/no question:
<input v-model="question" />
</p>
<p>{{ answer }}</p>
</div>
<!-- 因为 AJAX 库和通用工具的生态已经相当丰富,Vue 核心代码没有重复 -->
<!-- 提供这些功能以保持精简。这也可以让你自由选择自己更熟悉的工具。 -->
<script src="https://cdn.jsdelivr.net/npm/axios@0.12.0/dist/axios.min.js" rel="external nofollow" ></script>
<script>
const watchExampleVM = Vue.createApp({
data() {
return {
question: "",
answer: "Questions usually contain a question mark. ;-)"
}
},
watch: {
// whenever question changes, this function will run
question(newQuestion, oldQuestion) {
if (newQuestion.indexOf("?") > -1) {
this.getAnswer()
}
}
},
methods: {
getAnswer() {
this.answer = "Thinking..."
axios
.get("https://yesno.wtf/api")
.then(response => {
this.answer = response.data.answer
})
.catch(error => {
this.answer = "Error! Could not reach the API. " + error
})
}
}
}).mount("#watch-example")
</script>
结果: 点击此处实现
在这个示例中,使用 watch
选项允许我们执行异步操作 (访问一个 API),限制我们执行该操作的频率,并在我们得到最终结果前,设置中间状态。这些都是计算属性无法做到的。
除了 watch 选项之外,你还可以使用命令式的 vm.$watch API。
Vue 提供了一种更通用的方式来观察和响应当前活动的实例上的数据变动:侦听属性。当你有一些数据需要随着其它数据变动而变动时,你很容易滥用 watch
——特别是如果你之前使用过 AngularJS。然而,通常更好的做法是使用计算属性而不是命令式的 watch
回调。细想一下这个例子:
<div id="demo">{{ fullName }}</div>
const vm = Vue.createApp({
data() {
return {
firstName: "Foo",
lastName: "Bar",
fullName: "Foo Bar"
}
},
watch: {
firstName(val) {
this.fullName = val + " " + this.lastName
},
lastName(val) {
this.fullName = this.firstName + " " + val
}
}
}).mount("#demo")
上面代码是命令式且重复的。将它与计算属性的版本进行比较:
const vm = Vue.createApp({
data() {
return {
firstName: "Foo",
lastName: "Bar"
}
},
computed: {
fullName() {
return this.firstName + " " + this.lastName
}
}
}).mount("#demo")
好得多了,不是吗?
#v-ifv-if 指令用于条件性地渲染一块内容。这块内容只会在指令的表达式返回 truthy 值的时候被渲染。h1 v-if="awesome"Vue is aw...
#基础用法你可以用 v-model 指令在表单 input、textarea 及 select 元素上创建双向数据绑定。它会根据控件类型自动选取正确的方...
该页面假设你已经阅读过了组件基础。如果你还对组件不太了解,推荐你先阅读它。一个非 prop 的 attribute 是指传向一个组件,但...
该页面假设你已经阅读过了组件基础。如果你还对组件不太了解,推荐你先阅读它。#在动态组件上使用 keep-alive我们之前曾经在一个...
Vue 的过渡系统提供了非常多简单的方法设置进入、离开和列表的动效。那么对于数据元素本身的动效呢,比如:数字和运算颜色的显示...