作为“为大型前端项目”而设计的前端框架,Angular 其实有许多值得参考和学习的设计,本系列主要用于研究这些设计和功能的实现原理。本文主要围绕 Angular 中的 NgZone 的设计和实现来介绍。
上一篇我们介绍了 zone.js,它解决了很多 Javascript 异步编程时上下文的问题。
NgZone 基于 zone.js 集成了适用于 Angular 框架的一些能力。其中,对于 Angular 中的数据变更检测(脏检查)的性能优化,则主要依赖了 NgZone 的设计,我们一起来看一下。
NgZone
虽然 zone.js 可以监视同步和异步操作的所有状态,但 Angular 还提供了一项名为 NgZone 的服务。
NgZone 是一种用于在 Angular 区域内部或外部执行工作的可注射服务,对于不需要 Angular 处理 UI 更新或错误处理的异步任务来说,进行了性能优化的工作。
NgZone 设计
我们来看看 NgZone 的实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
| export class NgZone { readonly hasPendingMacrotasks: boolean = false; readonly hasPendingMicrotasks: boolean = false; readonly isStable: boolean = true; readonly onUnstable: EventEmitter<any> = new EventEmitter(false); readonly onMicrotaskEmpty: EventEmitter<any> = new EventEmitter(false); readonly onStable: EventEmitter<any> = new EventEmitter(false); readonly onError: EventEmitter<any> = new EventEmitter(false); constructor({ enableLongStackTrace = false, shouldCoalesceEventChangeDetection = false, shouldCoalesceRunChangeDetection = false }) { ... forkInnerZoneWithAngularBehavior(self); } static isInAngularZone(): boolean { return Zone.current.get('isAngularZone') === true; } run<T>(fn: (...args: any[]) => T, applyThis?: any, applyArgs?: any[]): T { return (this as any as NgZonePrivate)._inner.run(fn, applyThis, applyArgs); } runTask<T>(fn: (...args: any[]) => T, applyThis?: any, applyArgs?: any[], name?: string): T { const zone = (this as any as NgZonePrivate)._inner; const task = zone.scheduleEventTask('NgZoneEvent: ' + name, fn, EMPTY_PAYLOAD, noop, noop); try { return zone.runTask(task, applyThis, applyArgs); } finally { zone.cancelTask(task); } } runGuarded<T>(fn: (...args: any[]) => T, applyThis?: any, applyArgs?: any[]): T { return (this as any as NgZonePrivate)._inner.runGuarded(fn, applyThis, applyArgs); } runOutsideAngular<T>(fn: (...args: any[]) => T): T { return (this as any as NgZonePrivate)._outer.run(fn); } }
|
NgZone 基于 zone.js 之上再做了一层封装,通过fork
创建出子区域作为 Angular 区域:
1 2 3 4 5 6 7 8 9
| function forkInnerZoneWithAngularBehavior(zone: NgZonePrivate) { ... zone._inner = zone._inner.fork({ name: 'angular', properties: <any>{'isAngularZone': true}, ... }); }
|
除此之外,NgZone 里添加了用于表示没有微任务或宏任务的属性isStable
,可用于状态的检测。另外,NgZone 还定义了四个事件:
onUnstable
: 通知代码何时进入 Angular Zone,首先会在 VM Turn 上触发
onMicrotaskEmpty
: 通知当前的 VM Turn 中没有更多的微任务排队。这是 Angular 进行更改检测的提示,它可能会排队更多的微任务(此事件可在每次 VM 翻转时触发多次)
onStable
: 通知最后一个onMicrotaskEmpty
已运行并且没有更多的微任务,这意味着即将放弃 VM 转向(此事件仅被调用一次)
onError
: 通知已传送错误
上一节我们讲到,zone.js 处理了大多数异步 API,比如setTimeout()
、Promise.then()
和addEventListener()
等。对于一些 zone.js 无法处理的第三方 API,NgZone 服务的run()
方法可允许在 angular Zone 中执行函数。
通过使用 Angular Zone,函数中的所有异步操作会在正确的时间自动触发变更检测。
自动触发变更检测
当 NgZone 满足以下条件时,会创建一个名为 angular 的 Zone 来自动触发变更检测:
- 当执行同步或异步功能时(zone.js 内置变更检测,最终会通过
onMicrotaskEmpty
来触发)
- 已经没有已计划的 Microtask(
onMicrotaskEmpty
)
onMicrotaskEmpty
条件的触发监听,以及检测逻辑位于ApplicationRef
中:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| @Injectable() export class ApplicationRef { ... constructor( private _zone: NgZone, private _injector: Injector, private _exceptionHandler: ErrorHandler, private _componentFactoryResolver: ComponentFactoryResolver, private _initStatus: ApplicationInitStatus) { this._onMicrotaskEmptySubscription = this._zone.onMicrotaskEmpty.subscribe({ next: () => { this._zone.run(() => { this.tick(); }); } }); ... }
|
我们来看看,在什么时候会触发onMicrotaskEmpty
事件:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| function checkStable(zone: NgZonePrivate) { if (zone._nesting == 0 && !zone.hasPendingMicrotasks && !zone.isStable) { try { zone._nesting++; zone.onMicrotaskEmpty.emit(null); } finally { zone._nesting--; if (!zone.hasPendingMicrotasks) { try { zone.runOutsideAngular(() => zone.onStable.emit(null)); } finally { zone.isStable = true; } } } } }
|
当onInvokeTask
和onInvoke
两个钩子被触发时,微任务队列中可能会发生变化,因此 Angular 必须在每次钩子被触发时运行检查。除此之外,onHasTask
挂钩还用于执行检查,因为它跟踪整个队列更改:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
| function forkInnerZoneWithAngularBehavior(zone: NgZonePrivate) { const delayChangeDetectionForEventsDelegate = () => { delayChangeDetectionForEvents(zone); }; zone._inner = zone._inner.fork({ ... onInvokeTask: (delegate: ZoneDelegate, current: Zone, target: Zone, task: Task, applyThis: any, applyArgs: any): any => { ... delayChangeDetectionForEventsDelegate(); },
onInvoke: (delegate: ZoneDelegate, current: Zone, target: Zone, callback: Function, applyThis: any, applyArgs?: any[], source?: string): any => { ... delayChangeDetectionForEventsDelegate(); },
onHasTask: (delegate: ZoneDelegate, current: Zone, target: Zone, hasTaskState: HasTaskState) => { ... if (current === target) { if (hasTaskState.change == 'microTask') { zone._hasPendingMicrotasks = hasTaskState.microTask; updateMicroTaskStatus(zone); checkStable(zone); } ... } }, }); }
|
默认情况下,所有异步操作都在 Angular Zone 内,这会自动触发变更检测。
另一个常见的情况是我们不想触发变更检测(比如不希望像scroll
等事件过于频繁地进行变更检测,从而导致性能问题),此时可以使用 NgZone 的runOutsideAngular()
方法。
zone.js 能帮助 Angular 知道何时要触发变更检测,使得开发人员专注于应用开发。默认情况下,zone.js 已加载且无需其他配置即可工作。如果希望选择自己触发变更检测,则可以通过禁用 zone.js 的方式来处理。
总结
本文介绍了 NgZone 在 zone.js 的基础上进行了封装,从而使得在 Angular Zone 内函数中的所有异步操作可以在正确的时间自动触发变更检测。
可以根据自身的需要,使用 NgZone 的runOutsideAngular()
方法减少变更检测,也可以通过禁用 zone.js 的方式,来自己实现变更检测的逻辑。
参考
查看Github有更多内容噢:https://github.com/godbasin
更欢迎来被删的前端游乐场边撸猫边学前端噢
如果你想要关注日常生活中的我,欢迎关注“牧羊的猪”公众号噢