关于ChangeDetectorRef
,我已经知道 -
detectChanges
实际上触发了变化检测,而使用 -markForCheck
- 组件的实际变化检测未被调度,但是将来会发生(作为当前或下一个CD周期的一部分)
看看markForCheck
- 如果没有安排,那么什么时候会运行?显然是在Observables回调,异步回调和setTimout和事件之后。
The docs有OnPush
战略的组成部分
@Component({
selector: 'cmp',
changeDetection: ChangeDetectionStrategy.OnPush,
template: `Number of ticks: {{numberOfTicks}}`
})
class Cmp {
numberOfTicks = 0;
constructor(private ref: ChangeDetectorRef) {
setInterval(() => {
this.numberOfTicks++;
// the following is required, otherwise the view will not be updated
this.ref.markForCheck();
}, 1000);
}
}
题:
如果它标记为检查其祖先,在下一个周期,那么谁运行当前周期? (它是上一个setTimeout调用吗?)
因为这段代码显示每秒被另一个代替 - 换句话说每秒都有一个变化检测(?!)。
通过POV步骤实际发生了什么?
正如您所猜测的那样,markForCheck
,正如其名称所示,将告诉Angular标记要在下一个周期变化检测的组件。
当您编写setInterval时,这就是您实际编写的内容:
constructor(private ref: ChangeDetectorRef) {
setInterval(() => {
this.numberOfTicks++;
// the following is required, otherwise the view will not be updated
this.ref.markForCheck();
detectChangesFromZoneJS(); // this is a psudo code, but you can imagine something like this will happen, which is patched by NGZone
// Note that this is the last function that will always be called by Zone at the end of all the async events
}, 1000);
}
这由ZoneJS处理。
区域猴子修补了所有异步事件,当它们完成时,Zone会通知Angular,然后是Angular。知道是时候检测更改(根据最新的模型更改更新视图),但是当你是组件时,OnPush
不会检测到更改,除非组件内部发生特殊事情,(如点击事件) ,或任何@input
的更改)。
因此当你故意说markForCheck
时,你基本上会说:“我知道你不应该检测到这些变化,因为它只是OnPush,但我告诉你无论如何都要检测到它”
所以这是一步一步: