在使用 Angular 進行開發中,我們常用到 Angular 中的綁定——模型到視圖的輸入綁定、視圖到模型的輸出綁定以及視圖與模型的雙向綁定。而這些綁定的值之所以能在視圖與模型之間保持同步,正是得益於Angular中的變化檢測。 ...
什麼是變化監測
在使用 Angular 進行開發中,我們常用到 Angular 中的綁定——模型到視圖的輸入綁定、視圖到模型的輸出綁定以及視圖與模型的雙向綁定。而這些綁定的值之所以能在視圖與模型之間保持同步,正是得益於Angular中的變化檢測。
簡單來說,變化檢測就是 Angular 用來檢測視圖與模型之間綁定的值是否發生了改變,當檢測到模型中綁定的值發生改變時,則同步到視圖上,反之,當檢測到視圖上綁定的值發生改變時,則回調對應的綁定函數。
變化監測的源頭
變化監測的關鍵在於如何最小粒度地監測到綁定的值是否發生了改變,那麼在什麼情況下會導致這些綁定的值發生變化呢?我們可以看一下我們常用的幾種場景:
Events: click/hover/...
@Component({
selector: 'demo-component',
template: `
<h1>{{name}}</h1>
<button (click)="changeName()">change name</button>
`
})
export class DemoComponent {
name: string = 'Tom';
changeName() {
this.name = 'Jerry';
}
}
我們在模板中通過插值表達式綁定了 name 屬性。當點擊change name按鈕
時,改變了 name 屬性的值,這時模板視圖顯示內容也發生了改變。
XHR/webSocket
@Component({
selector: 'demo-component',
template: `
<h1>{{name}}</h1>
`
})
export class DemoComponent implements OnInit {
name: string = 'Tom';
constructor(public http: HttpClient) {}
ngOnInit() {
// 假設有這個./getNewName請求,返回一個新值'Jerry'
this.http.get('./getNewName').subscribe((data: string) => {
this.name = data;
});
}
}
我們在這個組件的 ngOnInit 函數里向伺服器端發送了一個 Ajax 請求,當這個請求返回結果時,同樣會改變當前模板視圖上綁定的 name 屬性的值。
Times: setTimeout/requestAnimationFrame
@Component({
selector: 'demo-component',
template: `
<h1>{{name}}</h1>
`
})
export class DemoComponent implements OnInit {
name: string = 'Tom';
constructor() {}
ngOnInit() {
// 假設有這個./getNewName請求,返回一個新值'Jerry'
setTimeout(() => {
this.name = 'Jerry';
}, 1000);
}
}
我們在這個組件的ngOnInit函數里通過設定一個定時任務,當定時任務執行時,同樣會改變當前視圖上綁定的name屬性的值。
總結
其實,我們不難發現上述三種情況都有一個共同點,即這些導致綁定值發生改變的事件都是非同步發生的。
Angular並不是捕捉對象的變動,它採用的是在適當的時機去檢驗對象的值是否被改動,這個時機就是這些非同步事件的發生。
這個時機是由 NgZone 這個服務去掌控的,它獲取到了整個應用的執行上下文,能夠對相關的非同步事件發生、完成或者異常等進行捕獲,然後驅動 Angular 的變化監測機制執行。
變化監測的處理機制
通過上面的介紹,我們大致明白了變化檢測是如何被觸發的,那麼 Angular 中的變化監測是如何執行的呢?
首先我們需要知道的是,對於每一個組件,都有一個對應的變化監測器;即每一個 Component 都對應有一個changeDetector
,我們可以在 Component 中通過依賴註入來獲取到changeDetector
。
而我們的多個 Component 是一個樹狀結構的組織,由於一個 Component 對應一個changeDetector
,那麼changeDetector
之間同樣是一個樹狀結構的組織。
最後我們需要記住的一點是,每次變化監測都是從 Component 樹根開始的。
舉個例子
子組件:
@Component({
selector: 'demo-child',
template: `
<h1>{{title}}</h1>
<p>{{paramOne}}</p>
<p>{{paramTwo}}</p>
`
})
export class DemoChildComponent {
title: string = '子組件標題';
@Input() paramOne: any; // 輸入屬性1
@Input() paramTwo: any; // 輸入屬性2
}
父組件:
@Component({
selector: 'demo-parent',
template: `
<h1>{{title}}</h1>
<demo-child [paramOne]='paramOneVal' [paramTwo]='paramTwoVal'></demo-child>
<button (click)="changeVal()">change name</button>
`
})
export class DemoParentComponent {
title: string = '父組件標題';
paramOneVal: any = '傳遞給paramOne的數據';
paramTwoVal: any = '傳遞給paramTwo的數據';
changeVal() {
this.paramOneVal = '改變之後的傳遞給paramOne的數據';
}
}
上面的代碼中,DemoParentComponent 通過 標簽嵌入了 DemoChildComponent,從樹狀結構上來說,DemoParentComponent 是 DemoChildComponent 的根節點,而 DemoChildComponent 是 DemoParentComponent 的葉子節點。
當我們點擊 DemoParentComponent 的 button 時,會回調到 changeVal 方法,然後會觸發變化監測的執行,變化監測流程如下:
首先變化檢測從 DemoParentComponent 開始:
檢測 title 值是否發生了變化:沒有發生變化
檢測 paramOneVal 值是否發生了變化:發生了變化(點擊按鈕調用changeVal()方法改變的)
檢測 paramTwoVal 值是否發生了變化:沒有發生變化
然後變化檢測進入到葉子節點 DemoChildComponent:
檢測 title 值是否發生了改變:沒有發生變化
檢測 paramOne 是否發生了變化:發生了改變(由於父組件的屬性paramOneVal發生了改變)
檢測 paramTwo 是否發生了改變:沒有發生變化
最後,因為 DemoChildComponent 再也沒有了葉子節點,所以變化監測將更新DOM,同步視圖與模型之間的變化。
變化監測策略
學習了變化監測的處理機制之後,你可能會想,這機制未免也有點太簡單粗暴了吧,假如我的應用中有成百上千個 Component,隨便一個 Component 觸發了監測,那麼都需要從根節點到葉子節點重新檢測一遍。
彆著急,Angular 的開發團隊已經考慮到了這個問題,上述的檢測機制只是一種預設的檢測機制,Angular 還提供一種 OnPush 的檢測機制(設置元數據屬性 changeDetection: ChangeDetectionStrategy.OnPush)。
OnPush 與 Default 之間的差別:當檢測到與子組件輸入綁定的值沒有發生改變時,變化檢測就不會深入到子組件中去。
變化監測類 - ChangeDetectorRef
上面說到我們可以修改組件元數據屬性 changeDetection 來修改組件的變化監測策略(ChangeDetectionStrategy.Default 或 ChangeDetectionStrategy.OnPush),除了這個,我們還可以使用 ChangeDetectorRef 來更加靈活的控制組件的變化監測。
Angular 在整個運行期間都會為每一個組件創建 ChangeDetectorRef 的實例,該實例提供了相關方法來手動管理變化監測。有了這個類,我們自己就可以自定義組件的變化監測策略了,如停止/啟用變化監測或者按指定路徑變化監測等等。
相關方法如下:
markForCheck():把根組件到該組件之間的這條路徑標記起來,通知Angular在下次觸發變化監測時必須檢查這條路徑上的組件。
detach():從變化監測樹中分離變化監測器,該組件的變化監測器將不再執行變化監測,除非再次手動執行reattach()方法。
reattach():把分離的變化監測器重新安裝上,使得該組件及其子組件都能執行變化監測。
detectChanges():手動觸發執行該組件到各個子組件的一次變化監測。
使用方法也很簡單,直接在組件中註入即可:
@Component({
selector: 'demo-parent',
template: `
<h1>{{title}}</h1>
`
})
export class DemoParentComponent implements OnInit {
title: string = '組件標題';
constructor(public cdRef: ChangeDetectorRef) {}
ngOnInit() {
this.cdRef.detach(); // 停止組件的變化監測,看需求使用不同的方法
}
}