Skip to content

Commit 8880b55

Browse files
committed
perf: improve unsub perf for deps with massive amount of subs
close #12696
1 parent 738f4b3 commit 8880b55

File tree

4 files changed

+43
-11
lines changed

4 files changed

+43
-11
lines changed

src/core/observer/dep.ts

+26-6
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,19 @@
1-
import { remove } from '../util/index'
21
import config from '../config'
32
import { DebuggerOptions, DebuggerEventExtraInfo } from 'v3'
43

54
let uid = 0
65

6+
const pendingCleanupDeps: Dep[] = []
7+
8+
export const cleanupDeps = () => {
9+
for (let i = 0; i < pendingCleanupDeps.length; i++) {
10+
const dep = pendingCleanupDeps[i]
11+
dep.subs = dep.subs.filter(s => s)
12+
dep._pending = false
13+
}
14+
pendingCleanupDeps.length = 0
15+
}
16+
717
/**
818
* @internal
919
*/
@@ -21,7 +31,9 @@ export interface DepTarget extends DebuggerOptions {
2131
export default class Dep {
2232
static target?: DepTarget | null
2333
id: number
24-
subs: Array<DepTarget>
34+
subs: Array<DepTarget | null>
35+
// pending subs cleanup
36+
_pending = false
2537

2638
constructor() {
2739
this.id = uid++
@@ -33,7 +45,15 @@ export default class Dep {
3345
}
3446

3547
removeSub(sub: DepTarget) {
36-
remove(this.subs, sub)
48+
// #12696 deps with massive amount of subscribers are extremely slow to
49+
// clean up in Chromium
50+
// to workaround this, we unset the sub for now, and clear them on
51+
// next scheduler flush.
52+
this.subs[this.subs.indexOf(sub)] = null
53+
if (!this._pending) {
54+
this._pending = true
55+
pendingCleanupDeps.push(this)
56+
}
3757
}
3858

3959
depend(info?: DebuggerEventExtraInfo) {
@@ -50,23 +70,23 @@ export default class Dep {
5070

5171
notify(info?: DebuggerEventExtraInfo) {
5272
// stabilize the subscriber list first
53-
const subs = this.subs.slice()
73+
const subs = this.subs.filter(s => s) as DepTarget[]
5474
if (__DEV__ && !config.async) {
5575
// subs aren't sorted in scheduler if not running async
5676
// we need to sort them now to make sure they fire in correct
5777
// order
5878
subs.sort((a, b) => a.id - b.id)
5979
}
6080
for (let i = 0, l = subs.length; i < l; i++) {
81+
const sub = subs[i]
6182
if (__DEV__ && info) {
62-
const sub = subs[i]
6383
sub.onTrigger &&
6484
sub.onTrigger({
6585
effect: subs[i],
6686
...info
6787
})
6888
}
69-
subs[i].update()
89+
sub.update()
7090
}
7191
}
7292
}

src/core/observer/scheduler.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import type Watcher from './watcher'
22
import config from '../config'
3-
import Dep from './dep'
3+
import Dep, { cleanupDeps } from './dep'
44
import { callHook, activateChildComponent } from '../instance/lifecycle'
55

66
import { warn, nextTick, devtools, inBrowser, isIE } from '../util/index'
@@ -121,6 +121,7 @@ function flushSchedulerQueue() {
121121
// call component updated and activated hooks
122122
callActivatedHooks(activatedQueue)
123123
callUpdatedHooks(updatedQueue)
124+
cleanupDeps()
124125

125126
// devtool hook
126127
/* istanbul ignore if */

src/shared/util.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -133,7 +133,13 @@ export const isReservedAttribute = makeMap('key,ref,slot,slot-scope,is')
133133
* Remove an item from an array.
134134
*/
135135
export function remove(arr: Array<any>, item: any): Array<any> | void {
136-
if (arr.length) {
136+
const len = arr.length
137+
if (len) {
138+
// fast path for the only / last item
139+
if (item === arr[len - 1]) {
140+
arr.length = len - 1
141+
return
142+
}
137143
const index = arr.indexOf(item)
138144
if (index > -1) {
139145
return arr.splice(index, 1)

test/unit/modules/observer/dep.spec.ts

+8-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import Dep from 'core/observer/dep'
1+
import Dep, { cleanupDeps } from 'core/observer/dep'
22

33
describe('Dep', () => {
44
let dep
@@ -24,8 +24,13 @@ describe('Dep', () => {
2424

2525
describe('removeSub()', () => {
2626
it('should remove sub', () => {
27-
dep.subs.push(null)
28-
dep.removeSub(null)
27+
const sub = {}
28+
dep.subs.push(sub)
29+
dep.removeSub(sub)
30+
expect(dep.subs.includes(sub)).toBe(false)
31+
32+
// nulled subs are cleared on next flush
33+
cleanupDeps()
2934
expect(dep.subs.length).toBe(0)
3035
})
3136
})

0 commit comments

Comments
 (0)