计算属性
双括弧的模板表达式是很便利的,但是当初设计他只是用来做简单的运算,如果在模板中放入大量的逻辑,会让模板过重,并且不好维护。
如:
<div id="example">
{{ message.split('').reverse().join('') }}
</div>
这里是修改了message的内容,虽然这样写也可以,但是有点违反了当初模板语法的设计思路,不推荐。
对于任何比较复杂的逻辑,推荐使用 计算属性。
<div id="example">
<p>Original message: "{{ message }}"</p>
<p>Computed reversed message: "{{ reversedMessage }}"</p>
</div>
var vm = new Vue({
el: '#example',
data: {
message: 'Hello'
},
computed: {
// 计算属性的 getter
reversedMessage: function () {
// `this` 指向 vm 实例
return this.message.split('').reverse().join('')
}
}
})
这里我们声明了一个计算属性 reversedMessage
。我们提供的函数将用作属性 vm.reversedMessage
的 getter
函数
reversedMessage
最后得到方法返回的修改过的值,插入到模板语法处。这也是响应式的,只要原来的message修改,就会触发计算属性。
计算属性缓存 vs 方法
刚才提到的 颠倒message的处理,我们通过methods中调用方法也一样可以做到。
<p>Reversed message: "{{ reversedMessage() }}"</p>
// 在组件中
methods: {
reversedMessage: function () {
return this.message.split('').reverse().join('')
}
}
我们可以将同一函数定义为一个方法而不是一个计算属性。两种方式的最终结果确实是完全相同的。
不同的地方是,计算属性是基于他们之间的依赖进行缓存的,只有在依赖关系或者值发生改变的时候才会重新请求,这就意味着只要 message
还没有发生改变,多次访问 reversedMessage
计算属性会立即返回之前的计算结果,而不必再次执行函数。
这样效率和性能也更加高,而且不需要手动触发。
相比方法,每次重新渲染,调用的方法总会执行的。
我们为什么需要缓存?假设我们有一个性能开销比较大的计算属性 A,它需要遍历一个巨大的数组并做大量的计算。然后我们可能有其他的计算属性依赖于 A 。如果没有缓存,我们将不可避免的多次执行 A 的 getter!如果你不希望有缓存,请用方法来替代。
计算属性 vs 监听属性
Vue提供了一种更通用的方式来观察和响应Vue实例上的数据变动,监听属性,当你的部分数据需要随着其他数据的变动而变动时,可容易 滥用watch
,特别是如果你之前使用过 angular,通常更好的方法是使用计算属性,而不是命令形式的 watch 回调。
div id="demo">{{ fullName }}</div>
var vm = new Vue({
el: '#demo',
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
}
}
})
上面代码是命令式且重复的,他通过监听2个值的变化来重新渲染新的值。将它与计算属性的版本进行比较:
var vm = new Vue({
el: '#demo',
data: {
firstName: 'Foo',
lastName: 'Bar'
},
computed: {
fullName: function () {
return this.firstName + ' ' + this.lastName
}
}
})
计算属性的setter
计算属性默认只有 getter,不过如果需要,可以添加一个 setter
// ...
computed: {
fullName: {
// getter
get: function () {
return this.firstName + ' ' + this.lastName
},
// setter
set: function (newValue) {
var names = newValue.split(' ')
this.firstName = names[0]
this.lastName = names[names.length - 1]
}
}
}
//
监听器
虽然计算属性在大多数情况下都比较合适,但有时候也是需要一个自定义的监听器。这就是为什么vue通过 watch 选项提供了一个更加通用的方法来响应数据的变化。当需要在数据变化时执行异步或者开销较大的操作时,这个时候使用 watch 是最有用的。
除了 watch
选项之外,您还可以使用命令式的 vm.$watch API。