Skip to content

Commit 16e06ca

Browse files
johnsoncodehksxzz
authored andcommitted
feat(reactivity): more efficient reactivity system (#5912)
fix #311, fix #1811, fix #6018, fix #7160, fix #8714, fix #9149, fix #9419, fix #9464
1 parent feb2f2e commit 16e06ca

23 files changed

+810
-542
lines changed

packages/reactivity/__tests__/computed.spec.ts

+164-1
Original file line numberDiff line numberDiff line change
@@ -184,7 +184,7 @@ describe('reactivity/computed', () => {
184184
// mutate n
185185
n.value++
186186
// on the 2nd run, plusOne.value should have already updated.
187-
expect(plusOneValues).toMatchObject([1, 2, 2])
187+
expect(plusOneValues).toMatchObject([1, 2])
188188
})
189189

190190
it('should warn if trying to set a readonly computed', () => {
@@ -288,4 +288,167 @@ describe('reactivity/computed', () => {
288288
oldValue: 2
289289
})
290290
})
291+
292+
// https://github.com/vuejs/core/pull/5912#issuecomment-1497596875
293+
it('should query deps dirty sequentially', () => {
294+
const cSpy = vi.fn()
295+
296+
const a = ref<null | { v: number }>({
297+
v: 1
298+
})
299+
const b = computed(() => {
300+
return a.value
301+
})
302+
const c = computed(() => {
303+
cSpy()
304+
return b.value?.v
305+
})
306+
const d = computed(() => {
307+
if (b.value) {
308+
return c.value
309+
}
310+
return 0
311+
})
312+
313+
d.value
314+
a.value!.v = 2
315+
a.value = null
316+
d.value
317+
expect(cSpy).toHaveBeenCalledTimes(1)
318+
})
319+
320+
// https://github.com/vuejs/core/pull/5912#issuecomment-1738257692
321+
it('chained computed dirty reallocation after querying dirty', () => {
322+
let _msg: string | undefined
323+
324+
const items = ref<number[]>()
325+
const isLoaded = computed(() => {
326+
return !!items.value
327+
})
328+
const msg = computed(() => {
329+
if (isLoaded.value) {
330+
return 'The items are loaded'
331+
} else {
332+
return 'The items are not loaded'
333+
}
334+
})
335+
336+
effect(() => {
337+
_msg = msg.value
338+
})
339+
340+
items.value = [1, 2, 3]
341+
items.value = [1, 2, 3]
342+
items.value = undefined
343+
344+
expect(_msg).toBe('The items are not loaded')
345+
})
346+
347+
it('chained computed dirty reallocation after trigger computed getter', () => {
348+
let _msg: string | undefined
349+
350+
const items = ref<number[]>()
351+
const isLoaded = computed(() => {
352+
return !!items.value
353+
})
354+
const msg = computed(() => {
355+
if (isLoaded.value) {
356+
return 'The items are loaded'
357+
} else {
358+
return 'The items are not loaded'
359+
}
360+
})
361+
362+
_msg = msg.value
363+
items.value = [1, 2, 3]
364+
isLoaded.value // <- trigger computed getter
365+
_msg = msg.value
366+
items.value = undefined
367+
_msg = msg.value
368+
369+
expect(_msg).toBe('The items are not loaded')
370+
})
371+
372+
// https://github.com/vuejs/core/pull/5912#issuecomment-1739159832
373+
it('deps order should be consistent with the last time get value', () => {
374+
const cSpy = vi.fn()
375+
376+
const a = ref(0)
377+
const b = computed(() => {
378+
return a.value % 3 !== 0
379+
})
380+
const c = computed(() => {
381+
cSpy()
382+
if (a.value % 3 === 2) {
383+
return 'expensive'
384+
}
385+
return 'cheap'
386+
})
387+
const d = computed(() => {
388+
return a.value % 3 === 2
389+
})
390+
const e = computed(() => {
391+
if (b.value) {
392+
if (d.value) {
393+
return 'Avoiding expensive calculation'
394+
}
395+
}
396+
return c.value
397+
})
398+
399+
e.value
400+
a.value++
401+
e.value
402+
403+
expect(e.effect.deps.length).toBe(3)
404+
expect(e.effect.deps.indexOf((b as any).dep)).toBe(0)
405+
expect(e.effect.deps.indexOf((d as any).dep)).toBe(1)
406+
expect(e.effect.deps.indexOf((c as any).dep)).toBe(2)
407+
expect(cSpy).toHaveBeenCalledTimes(2)
408+
409+
a.value++
410+
e.value
411+
412+
expect(cSpy).toHaveBeenCalledTimes(2)
413+
})
414+
415+
it('should trigger by the second computed that maybe dirty', () => {
416+
const cSpy = vi.fn()
417+
418+
const src1 = ref(0)
419+
const src2 = ref(0)
420+
const c1 = computed(() => src1.value)
421+
const c2 = computed(() => (src1.value % 2) + src2.value)
422+
const c3 = computed(() => {
423+
cSpy()
424+
c1.value
425+
c2.value
426+
})
427+
428+
c3.value
429+
src1.value = 2
430+
c3.value
431+
expect(cSpy).toHaveBeenCalledTimes(2)
432+
src2.value = 1
433+
c3.value
434+
expect(cSpy).toHaveBeenCalledTimes(3)
435+
})
436+
437+
it('should trigger the second effect', () => {
438+
const fnSpy = vi.fn()
439+
const v = ref(1)
440+
const c = computed(() => v.value)
441+
442+
effect(() => {
443+
c.value
444+
})
445+
effect(() => {
446+
c.value
447+
fnSpy()
448+
})
449+
450+
expect(fnSpy).toBeCalledTimes(1)
451+
v.value = 2
452+
expect(fnSpy).toBeCalledTimes(2)
453+
})
291454
})
Original file line numberDiff line numberDiff line change
@@ -1,57 +1,32 @@
1-
import { computed, deferredComputed, effect, ref } from '../src'
1+
import { computed, effect, ref } from '../src'
22

33
describe('deferred computed', () => {
4-
const tick = Promise.resolve()
5-
6-
test('should only trigger once on multiple mutations', async () => {
4+
test('should not trigger if value did not change', () => {
75
const src = ref(0)
8-
const c = deferredComputed(() => src.value)
6+
const c = computed(() => src.value % 2)
97
const spy = vi.fn()
108
effect(() => {
119
spy(c.value)
1210
})
1311
expect(spy).toHaveBeenCalledTimes(1)
14-
src.value = 1
1512
src.value = 2
16-
src.value = 3
17-
// not called yet
18-
expect(spy).toHaveBeenCalledTimes(1)
19-
await tick
20-
// should only trigger once
21-
expect(spy).toHaveBeenCalledTimes(2)
22-
expect(spy).toHaveBeenCalledWith(c.value)
23-
})
2413

25-
test('should not trigger if value did not change', async () => {
26-
const src = ref(0)
27-
const c = deferredComputed(() => src.value % 2)
28-
const spy = vi.fn()
29-
effect(() => {
30-
spy(c.value)
31-
})
32-
expect(spy).toHaveBeenCalledTimes(1)
33-
src.value = 1
34-
src.value = 2
35-
36-
await tick
3714
// should not trigger
3815
expect(spy).toHaveBeenCalledTimes(1)
3916

4017
src.value = 3
41-
src.value = 4
4218
src.value = 5
43-
await tick
4419
// should trigger because latest value changes
4520
expect(spy).toHaveBeenCalledTimes(2)
4621
})
4722

48-
test('chained computed trigger', async () => {
23+
test('chained computed trigger', () => {
4924
const effectSpy = vi.fn()
5025
const c1Spy = vi.fn()
5126
const c2Spy = vi.fn()
5227

5328
const src = ref(0)
54-
const c1 = deferredComputed(() => {
29+
const c1 = computed(() => {
5530
c1Spy()
5631
return src.value % 2
5732
})
@@ -69,19 +44,18 @@ describe('deferred computed', () => {
6944
expect(effectSpy).toHaveBeenCalledTimes(1)
7045

7146
src.value = 1
72-
await tick
7347
expect(c1Spy).toHaveBeenCalledTimes(2)
7448
expect(c2Spy).toHaveBeenCalledTimes(2)
7549
expect(effectSpy).toHaveBeenCalledTimes(2)
7650
})
7751

78-
test('chained computed avoid re-compute', async () => {
52+
test('chained computed avoid re-compute', () => {
7953
const effectSpy = vi.fn()
8054
const c1Spy = vi.fn()
8155
const c2Spy = vi.fn()
8256

8357
const src = ref(0)
84-
const c1 = deferredComputed(() => {
58+
const c1 = computed(() => {
8559
c1Spy()
8660
return src.value % 2
8761
})
@@ -98,26 +72,24 @@ describe('deferred computed', () => {
9872
src.value = 2
9973
src.value = 4
10074
src.value = 6
101-
await tick
102-
// c1 should re-compute once.
103-
expect(c1Spy).toHaveBeenCalledTimes(2)
75+
expect(c1Spy).toHaveBeenCalledTimes(4)
10476
// c2 should not have to re-compute because c1 did not change.
10577
expect(c2Spy).toHaveBeenCalledTimes(1)
10678
// effect should not trigger because c2 did not change.
10779
expect(effectSpy).toHaveBeenCalledTimes(1)
10880
})
10981

110-
test('chained computed value invalidation', async () => {
82+
test('chained computed value invalidation', () => {
11183
const effectSpy = vi.fn()
11284
const c1Spy = vi.fn()
11385
const c2Spy = vi.fn()
11486

11587
const src = ref(0)
116-
const c1 = deferredComputed(() => {
88+
const c1 = computed(() => {
11789
c1Spy()
11890
return src.value % 2
11991
})
120-
const c2 = deferredComputed(() => {
92+
const c2 = computed(() => {
12193
c2Spy()
12294
return c1.value + 1
12395
})
@@ -139,17 +111,17 @@ describe('deferred computed', () => {
139111
expect(c2Spy).toHaveBeenCalledTimes(2)
140112
})
141113

142-
test('sync access of invalidated chained computed should not prevent final effect from running', async () => {
114+
test('sync access of invalidated chained computed should not prevent final effect from running', () => {
143115
const effectSpy = vi.fn()
144116
const c1Spy = vi.fn()
145117
const c2Spy = vi.fn()
146118

147119
const src = ref(0)
148-
const c1 = deferredComputed(() => {
120+
const c1 = computed(() => {
149121
c1Spy()
150122
return src.value % 2
151123
})
152-
const c2 = deferredComputed(() => {
124+
const c2 = computed(() => {
153125
c2Spy()
154126
return c1.value + 1
155127
})
@@ -162,14 +134,13 @@ describe('deferred computed', () => {
162134
src.value = 1
163135
// sync access c2
164136
c2.value
165-
await tick
166137
expect(effectSpy).toHaveBeenCalledTimes(2)
167138
})
168139

169-
test('should not compute if deactivated before scheduler is called', async () => {
140+
test('should not compute if deactivated before scheduler is called', () => {
170141
const c1Spy = vi.fn()
171142
const src = ref(0)
172-
const c1 = deferredComputed(() => {
143+
const c1 = computed(() => {
173144
c1Spy()
174145
return src.value % 2
175146
})
@@ -179,7 +150,6 @@ describe('deferred computed', () => {
179150
c1.effect.stop()
180151
// trigger
181152
src.value++
182-
await tick
183153
expect(c1Spy).toHaveBeenCalledTimes(1)
184154
})
185155
})

0 commit comments

Comments
 (0)