計算屬性
computed
計算緩存 vs Methods
<div id="example"> <p>Original message: "{{ message }}"</p> <p>Computed reversed message: "{{ reversedMessage }}"</p> </div>
js
var vm = new Vue({ el: '#example', data: { message: 'Hello' }, computed: { // a computed getter reversedMessage: function () { // `this` points to the vm instance return this.message.split('').reverse().join('') } } })
這里我們聲明了一個計算屬性reverseMessage。我們提供的函數將用作屬性vm.reversedMessage 的getter。
你可以像綁定普通屬性一樣在模板中綁定計算屬性。Vue知道vm.reversedMessage 依賴於vm.message,因此當
vm.message 發生改變時,所有依賴於vm.reversedMessage的綁定也會更新。而且最妙的是我們已經以聲明的方式
創建了這種依賴關系:計算屬性的getter是沒有副作用,這使得它易於測試和推理。
計算緩存 vs Methods
我們可以將同一個函數定義為一個method而不是一個計算屬性。對於最終的結果,兩種方式確實是相同的。然而,不同的是計算屬性是基於
他們的依賴進行緩存的。計算屬性只有在它的相關依賴發生改變時才會重新求值。這就意味着只要它的依賴message沒有發生改變,多次訪問
veversedMessage計算屬性會立即返回之前的計算結果,而不必再次執行函數。
相比而言,只要發生重新渲染,method調用總會執行該函數。
我們為什么需要緩存?假設我們有一個性能開銷比較大的的計算屬性 A ,它需要遍歷一個極大的數組和做大量的計算。然后我們可能有其他的計算屬性依賴於 A 。如果沒有緩存,我們將不可避免的多次執行 A 的 getter!如果你不希望有緩存,請用 method 替代。
Vue 確實提供了一種更通用的方式來觀察和響應 Vue 實例上的數據變動:watch 屬性。當你有一些數據需要隨着其它數據變動而變動時,你很容易濫用 watch
——特別是如果你之前使用過 AngularJS。然而,通常更好的想法是使用 computed 屬性而不是命令式的 watch
回調。細想一下這個例子:
computed屬性 vs watched 屬性
<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 } } })
上面代碼是命令式的和重復的。將它與 computed 屬性的版本進行比較:
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] } } } // ...
現在在運行 vm.fullName = 'John Doe'
時, setter 會被調用, vm.firstName
和 vm.lastName
也相應地會被更新。