Skip to content

Commit 653aac2

Browse files
authored
perf: avoid unnecessary re-renders when computed property value did not change (#7824)
close #7767
1 parent f43ce3a commit 653aac2

File tree

4 files changed

+153
-51
lines changed

4 files changed

+153
-51
lines changed

src/core/instance/state.js

+4-9
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
import config from '../config'
44
import Watcher from '../observer/watcher'
5-
import Dep, { pushTarget, popTarget } from '../observer/dep'
5+
import { pushTarget, popTarget } from '../observer/dep'
66
import { isUpdatingChildComponent } from './lifecycle'
77

88
import {
@@ -164,7 +164,7 @@ export function getData (data: Function, vm: Component): any {
164164
}
165165
}
166166

167-
const computedWatcherOptions = { lazy: true }
167+
const computedWatcherOptions = { computed: true }
168168

169169
function initComputed (vm: Component, computed: Object) {
170170
// $flow-disable-line
@@ -244,13 +244,8 @@ function createComputedGetter (key) {
244244
return function computedGetter () {
245245
const watcher = this._computedWatchers && this._computedWatchers[key]
246246
if (watcher) {
247-
if (watcher.dirty) {
248-
watcher.evaluate()
249-
}
250-
if (Dep.target) {
251-
watcher.depend()
252-
}
253-
return watcher.value
247+
watcher.depend()
248+
return watcher.evaluate()
254249
}
255250
}
256251
}

src/core/observer/watcher.js

+64-37
Original file line numberDiff line numberDiff line change
@@ -29,10 +29,11 @@ export default class Watcher {
2929
id: number;
3030
deep: boolean;
3131
user: boolean;
32-
lazy: boolean;
32+
computed: boolean;
3333
sync: boolean;
3434
dirty: boolean;
3535
active: boolean;
36+
dep: Dep;
3637
deps: Array<Dep>;
3738
newDeps: Array<Dep>;
3839
depIds: SimpleSet;
@@ -57,16 +58,16 @@ export default class Watcher {
5758
if (options) {
5859
this.deep = !!options.deep
5960
this.user = !!options.user
60-
this.lazy = !!options.lazy
61+
this.computed = !!options.computed
6162
this.sync = !!options.sync
6263
this.before = options.before
6364
} else {
64-
this.deep = this.user = this.lazy = this.sync = false
65+
this.deep = this.user = this.computed = this.sync = false
6566
}
6667
this.cb = cb
6768
this.id = ++uid // uid for batching
6869
this.active = true
69-
this.dirty = this.lazy // for lazy watchers
70+
this.dirty = this.computed // for computed watchers
7071
this.deps = []
7172
this.newDeps = []
7273
this.depIds = new Set()
@@ -89,9 +90,12 @@ export default class Watcher {
8990
)
9091
}
9192
}
92-
this.value = this.lazy
93-
? undefined
94-
: this.get()
93+
if (this.computed) {
94+
this.value = undefined
95+
this.dep = new Dep()
96+
} else {
97+
this.value = this.get()
98+
}
9599
}
96100

97101
/**
@@ -162,8 +166,24 @@ export default class Watcher {
162166
*/
163167
update () {
164168
/* istanbul ignore else */
165-
if (this.lazy) {
166-
this.dirty = true
169+
if (this.computed) {
170+
// A computed property watcher has two modes: lazy and activated.
171+
// It initializes as lazy by default, and only becomes activated when
172+
// it is depended on by at least one subscriber, which is typically
173+
// another computed property or a component's render function.
174+
if (this.dep.subs.length === 0) {
175+
// In lazy mode, we don't want to perform computations until necessary,
176+
// so we simply mark the watcher as dirty. The actual computation is
177+
// performed just-in-time in this.evaluate() when the computed property
178+
// is accessed.
179+
this.dirty = true
180+
} else {
181+
// In activated mode, we want to proactively perform the computation
182+
// but only notify our subscribers when the value has indeed changed.
183+
this.getAndInvoke(() => {
184+
this.dep.notify()
185+
})
186+
}
167187
} else if (this.sync) {
168188
this.run()
169189
} else {
@@ -177,47 +197,54 @@ export default class Watcher {
177197
*/
178198
run () {
179199
if (this.active) {
180-
const value = this.get()
181-
if (
182-
value !== this.value ||
183-
// Deep watchers and watchers on Object/Arrays should fire even
184-
// when the value is the same, because the value may
185-
// have mutated.
186-
isObject(value) ||
187-
this.deep
188-
) {
189-
// set new value
190-
const oldValue = this.value
191-
this.value = value
192-
if (this.user) {
193-
try {
194-
this.cb.call(this.vm, value, oldValue)
195-
} catch (e) {
196-
handleError(e, this.vm, `callback for watcher "${this.expression}"`)
197-
}
198-
} else {
199-
this.cb.call(this.vm, value, oldValue)
200+
this.getAndInvoke(this.cb)
201+
}
202+
}
203+
204+
getAndInvoke (cb: Function) {
205+
const value = this.get()
206+
if (
207+
value !== this.value ||
208+
// Deep watchers and watchers on Object/Arrays should fire even
209+
// when the value is the same, because the value may
210+
// have mutated.
211+
isObject(value) ||
212+
this.deep
213+
) {
214+
// set new value
215+
const oldValue = this.value
216+
this.value = value
217+
this.dirty = false
218+
if (this.user) {
219+
try {
220+
cb.call(this.vm, value, oldValue)
221+
} catch (e) {
222+
handleError(e, this.vm, `callback for watcher "${this.expression}"`)
200223
}
224+
} else {
225+
cb.call(this.vm, value, oldValue)
201226
}
202227
}
203228
}
204229

205230
/**
206-
* Evaluate the value of the watcher.
207-
* This only gets called for lazy watchers.
231+
* Evaluate and return the value of the watcher.
232+
* This only gets called for computed property watchers.
208233
*/
209234
evaluate () {
210-
this.value = this.get()
211-
this.dirty = false
235+
if (this.dirty) {
236+
this.value = this.get()
237+
this.dirty = false
238+
}
239+
return this.value
212240
}
213241

214242
/**
215-
* Depend on all deps collected by this watcher.
243+
* Depend on this watcher. Only for computed property watchers.
216244
*/
217245
depend () {
218-
let i = this.deps.length
219-
while (i--) {
220-
this.deps[i].depend()
246+
if (this.dep && Dep.target) {
247+
this.dep.depend()
221248
}
222249
}
223250

test/unit/features/options/computed.spec.js

+36
Original file line numberDiff line numberDiff line change
@@ -216,4 +216,40 @@ describe('Options computed', () => {
216216
})
217217
expect(() => vm.a).toThrowError('rethrow')
218218
})
219+
220+
// #7767
221+
it('should avoid unnecessary re-renders', done => {
222+
const computedSpy = jasmine.createSpy('computed')
223+
const updatedSpy = jasmine.createSpy('updated')
224+
const vm = new Vue({
225+
data: {
226+
msg: 'bar'
227+
},
228+
computed: {
229+
a () {
230+
computedSpy()
231+
return this.msg !== 'foo'
232+
}
233+
},
234+
template: `<div>{{ a }}</div>`,
235+
updated: updatedSpy
236+
}).$mount()
237+
238+
expect(vm.$el.textContent).toBe('true')
239+
expect(computedSpy.calls.count()).toBe(1)
240+
expect(updatedSpy.calls.count()).toBe(0)
241+
242+
vm.msg = 'baz'
243+
waitForUpdate(() => {
244+
expect(vm.$el.textContent).toBe('true')
245+
expect(computedSpy.calls.count()).toBe(2)
246+
expect(updatedSpy.calls.count()).toBe(0)
247+
}).then(() => {
248+
vm.msg = 'foo'
249+
}).then(() => {
250+
expect(vm.$el.textContent).toBe('false')
251+
expect(computedSpy.calls.count()).toBe(3)
252+
expect(updatedSpy.calls.count()).toBe(1)
253+
}).then(done)
254+
})
219255
})

test/unit/modules/observer/watcher.spec.js

+49-5
Original file line numberDiff line numberDiff line change
@@ -144,26 +144,70 @@ describe('Watcher', () => {
144144
}).then(done)
145145
})
146146

147-
it('lazy mode', done => {
147+
it('computed mode, lazy', done => {
148+
let getterCallCount = 0
148149
const watcher = new Watcher(vm, function () {
150+
getterCallCount++
149151
return this.a + this.b.d
150-
}, null, { lazy: true })
151-
expect(watcher.lazy).toBe(true)
152+
}, null, { computed: true })
153+
154+
expect(getterCallCount).toBe(0)
155+
expect(watcher.computed).toBe(true)
152156
expect(watcher.value).toBeUndefined()
153157
expect(watcher.dirty).toBe(true)
154-
watcher.evaluate()
158+
expect(watcher.dep).toBeTruthy()
159+
160+
const value = watcher.evaluate()
161+
expect(getterCallCount).toBe(1)
162+
expect(value).toBe(5)
155163
expect(watcher.value).toBe(5)
156164
expect(watcher.dirty).toBe(false)
165+
166+
// should not get again if not dirty
167+
watcher.evaluate()
168+
expect(getterCallCount).toBe(1)
169+
157170
vm.a = 2
158171
waitForUpdate(() => {
172+
expect(getterCallCount).toBe(1)
159173
expect(watcher.value).toBe(5)
160174
expect(watcher.dirty).toBe(true)
161-
watcher.evaluate()
175+
176+
const value = watcher.evaluate()
177+
expect(getterCallCount).toBe(2)
178+
expect(value).toBe(6)
162179
expect(watcher.value).toBe(6)
163180
expect(watcher.dirty).toBe(false)
164181
}).then(done)
165182
})
166183

184+
it('computed mode, activated', done => {
185+
let getterCallCount = 0
186+
const watcher = new Watcher(vm, function () {
187+
getterCallCount++
188+
return this.a + this.b.d
189+
}, null, { computed: true })
190+
191+
// activate by mocking a subscriber
192+
const subMock = jasmine.createSpyObj('sub', ['update'])
193+
watcher.dep.addSub(subMock)
194+
195+
const value = watcher.evaluate()
196+
expect(getterCallCount).toBe(1)
197+
expect(value).toBe(5)
198+
199+
vm.a = 2
200+
waitForUpdate(() => {
201+
expect(getterCallCount).toBe(2)
202+
expect(subMock.update).toHaveBeenCalled()
203+
204+
// since already computed, calling evaluate again should not trigger
205+
// getter
206+
watcher.evaluate()
207+
expect(getterCallCount).toBe(2)
208+
}).then(done)
209+
})
210+
167211
it('teardown', done => {
168212
const watcher = new Watcher(vm, 'b.c', spy)
169213
watcher.teardown()

0 commit comments

Comments
 (0)