Skip to content

Commit 6b1d431

Browse files
committed
Revert "perf: avoid unnecessary re-renders when computed property value did not change (#7824)"
This reverts commit 653aac2.
1 parent 2ae4e29 commit 6b1d431

File tree

4 files changed

+51
-153
lines changed

4 files changed

+51
-153
lines changed

src/core/instance/state.js

+9-4
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 { pushTarget, popTarget } from '../observer/dep'
5+
import Dep, { 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 = { computed: true }
167+
const computedWatcherOptions = { lazy: true }
168168

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

src/core/observer/watcher.js

+37-64
Original file line numberDiff line numberDiff line change
@@ -29,11 +29,10 @@ export default class Watcher {
2929
id: number;
3030
deep: boolean;
3131
user: boolean;
32-
computed: boolean;
32+
lazy: boolean;
3333
sync: boolean;
3434
dirty: boolean;
3535
active: boolean;
36-
dep: Dep;
3736
deps: Array<Dep>;
3837
newDeps: Array<Dep>;
3938
depIds: SimpleSet;
@@ -58,16 +57,16 @@ export default class Watcher {
5857
if (options) {
5958
this.deep = !!options.deep
6059
this.user = !!options.user
61-
this.computed = !!options.computed
60+
this.lazy = !!options.lazy
6261
this.sync = !!options.sync
6362
this.before = options.before
6463
} else {
65-
this.deep = this.user = this.computed = this.sync = false
64+
this.deep = this.user = this.lazy = this.sync = false
6665
}
6766
this.cb = cb
6867
this.id = ++uid // uid for batching
6968
this.active = true
70-
this.dirty = this.computed // for computed watchers
69+
this.dirty = this.lazy // for lazy watchers
7170
this.deps = []
7271
this.newDeps = []
7372
this.depIds = new Set()
@@ -90,12 +89,9 @@ export default class Watcher {
9089
)
9190
}
9291
}
93-
if (this.computed) {
94-
this.value = undefined
95-
this.dep = new Dep()
96-
} else {
97-
this.value = this.get()
98-
}
92+
this.value = this.lazy
93+
? undefined
94+
: this.get()
9995
}
10096

10197
/**
@@ -166,24 +162,8 @@ export default class Watcher {
166162
*/
167163
update () {
168164
/* istanbul ignore else */
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-
}
165+
if (this.lazy) {
166+
this.dirty = true
187167
} else if (this.sync) {
188168
this.run()
189169
} else {
@@ -197,54 +177,47 @@ export default class Watcher {
197177
*/
198178
run () {
199179
if (this.active) {
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}"`)
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)
223200
}
224-
} else {
225-
cb.call(this.vm, value, oldValue)
226201
}
227202
}
228203
}
229204

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

242214
/**
243-
* Depend on this watcher. Only for computed property watchers.
215+
* Depend on all deps collected by this watcher.
244216
*/
245217
depend () {
246-
if (this.dep && Dep.target) {
247-
this.dep.depend()
218+
let i = this.deps.length
219+
while (i--) {
220+
this.deps[i].depend()
248221
}
249222
}
250223

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

-36
Original file line numberDiff line numberDiff line change
@@ -216,40 +216,4 @@ 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-
})
255219
})

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

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

147-
it('computed mode, lazy', done => {
148-
let getterCallCount = 0
147+
it('lazy mode', done => {
149148
const watcher = new Watcher(vm, function () {
150-
getterCallCount++
151149
return this.a + this.b.d
152-
}, null, { computed: true })
153-
154-
expect(getterCallCount).toBe(0)
155-
expect(watcher.computed).toBe(true)
150+
}, null, { lazy: true })
151+
expect(watcher.lazy).toBe(true)
156152
expect(watcher.value).toBeUndefined()
157153
expect(watcher.dirty).toBe(true)
158-
expect(watcher.dep).toBeTruthy()
159-
160-
const value = watcher.evaluate()
161-
expect(getterCallCount).toBe(1)
162-
expect(value).toBe(5)
154+
watcher.evaluate()
163155
expect(watcher.value).toBe(5)
164156
expect(watcher.dirty).toBe(false)
165-
166-
// should not get again if not dirty
167-
watcher.evaluate()
168-
expect(getterCallCount).toBe(1)
169-
170157
vm.a = 2
171158
waitForUpdate(() => {
172-
expect(getterCallCount).toBe(1)
173159
expect(watcher.value).toBe(5)
174160
expect(watcher.dirty).toBe(true)
175-
176-
const value = watcher.evaluate()
177-
expect(getterCallCount).toBe(2)
178-
expect(value).toBe(6)
161+
watcher.evaluate()
179162
expect(watcher.value).toBe(6)
180163
expect(watcher.dirty).toBe(false)
181164
}).then(done)
182165
})
183166

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-
211167
it('teardown', done => {
212168
const watcher = new Watcher(vm, 'b.c', spy)
213169
watcher.teardown()

0 commit comments

Comments
 (0)