Skip to content

Commit 1a526f2

Browse files
atscottdylhunn
authored andcommitted
perf(common): AsyncPipe should not call markForCheck on subscription (#54554)
This commit prevents `AsyncPipe` from calling `markForCheck` when values are synchronously emit during subscription to an observable. This prevents subscriptions to `Replay` observables from needlessly walking up to the root of the view tree during template execution for each new replay observable in the template. PR Close #54554
1 parent 600a8cd commit 1a526f2

File tree

1 file changed

+13
-4
lines changed

1 file changed

+13
-4
lines changed

packages/common/src/pipes/async_pipe.ts

+13-4
Original file line numberDiff line numberDiff line change
@@ -102,6 +102,7 @@ const _subscribableStrategy = new SubscribableStrategy();
102102
export class AsyncPipe implements OnDestroy, PipeTransform {
103103
private _ref: ChangeDetectorRef | null;
104104
private _latestValue: any = null;
105+
private markForCheckOnValueUpdate = true;
105106

106107
private _subscription: Unsubscribable | Promise<any> | null = null;
107108
private _obj: Subscribable<any> | Promise<any> | EventEmitter<any> | null = null;
@@ -134,7 +135,15 @@ export class AsyncPipe implements OnDestroy, PipeTransform {
134135
transform<T>(obj: Observable<T> | Subscribable<T> | Promise<T> | null | undefined): T | null {
135136
if (!this._obj) {
136137
if (obj) {
137-
this._subscribe(obj);
138+
try {
139+
// Only call `markForCheck` if the value is updated asynchronously.
140+
// Synchronous updates _during_ subscription should not wastefully mark for check -
141+
// this value is already going to be returned from the transform function.
142+
this.markForCheckOnValueUpdate = false;
143+
this._subscribe(obj);
144+
} finally {
145+
this.markForCheckOnValueUpdate = true;
146+
}
138147
}
139148
return this._latestValue;
140149
}
@@ -181,9 +190,9 @@ export class AsyncPipe implements OnDestroy, PipeTransform {
181190
private _updateLatestValue(async: any, value: Object): void {
182191
if (async === this._obj) {
183192
this._latestValue = value;
184-
// Note: `this._ref` is only cleared in `ngOnDestroy` so is known to be available when a
185-
// value is being updated.
186-
this._ref!.markForCheck();
193+
if (this.markForCheckOnValueUpdate) {
194+
this._ref?.markForCheck();
195+
}
187196
}
188197
}
189198
}

0 commit comments

Comments
 (0)