欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Angular2學習教程之ng中變更檢測問題詳解

 更新時間:2017年05月28日 16:50:46   作者:panode  
這篇文章主要給大家介紹了Angular2學習教程之ng中變更檢測問題的相關資料,文中介紹的非常詳細,對大家具有一定的參考學習價值,需要的朋友們下面來一起看看吧。

開發(fā)中遇到的問題

在開發(fā)中遇到一個這樣的問題,代碼不便透露,這里用簡單的例子還原一下問題所在:

有三個組件,第一個是用來展示Todo列表的組件TodoComponent,Todo是個類,包含id和name屬性。

@Component({
 selector: 'todo-list',
 template: `
  <p *ngFor='let item of todos'>{{ item.name }}</p>
 `,
})
export class TodoComponent{
 @Input() todos: Todo[];

 public getTodos():Todo[]{
  return this.todos;
 }
}

第二個組件同樣是一個Todo列表展示組件TodoDataComponent ,不同的是該組件需要一個TodoComponent類型的輸入,并從TodoComponent組件中獲得需要展示的Todo數(shù)據(jù)。

@Component({
 selector: 'app-todo-data',
 template: `<p *ngFor='let item of todos'>{{ item.name }}</p>
  <button (click)='getData()'>get data</button>`,
 styleUrls: ['./todo-data.component.css'],
 inputs: ['todoComponent'],
})
export class TodoDataComponent implements OnInit {
 todoComponent: TodoComponent;
 todos: Todo[]
 constructor() { }
 ngOnInit() {
 }
 getData(){
  this.todos=this.todoComponent.getTodos();
 }
}

最后一個是應用的根組件,根組件根據(jù)loading值來確定是否加載TodoComponent組件,并展示TodoDataComponent 組件。

//app.component.htm
<div>
 <div *ngIf='loading'>
  <todo-list [todos]='todos'></todo-list>
  <button (click)='changeall()'>next</button>
 </div>
</div>
<div>
 <app-todo-data [todoComponent]='todoComponent'></app-todo-data>
</div>

//app.component.ts
@Component({
 selector: 'app-root',
 templateUrl: './app.component.html',
 styleUrls: ['./app.component.css'],
})
export class AppComponent implements OnInit {
 todos: Todo[];
 @ViewChild(TodoComponent)
 todoComponent: TodoComponent;
 loading: boolean = true;
 constructor(private todoService:TodoService){
  super(true);
 }
 ngOnInit(){
  this.todoService.todos.subscribe(data => {this.todos=data});
  this.todoService.load(0, 3);
 }
 changeall(){
  this.todoService.load(3, 3);
 }
}

這樣問題就來了,TodoComponent 組件是否在頁面上展示是不確定的,在上面的例子中根組件最開始沒有渲染TodoComponent組件,最后根據(jù)loading的值將TodoComponent渲染出來。而TodoDataComponent 組件的顯示又需要一個TodoComponent 進行初始化(跟組件通過@ViewChild(TodoComponent)獲得),這樣造成在開發(fā)模式下出現(xiàn)以下錯誤:
template:9:16 caused by: Expression has changed after it was checked. Previous value: 'undefined'. Current value: '[object Object]'.

該錯誤僅在開發(fā)模式下會報告出來的,解決掉總是更好的選擇,防止在生產(chǎn)環(huán)境下出現(xiàn)問題。

問題的原因及解決辦法

這個問題是ng2中的變更檢測策略造成的,ng2并沒有智能到一有數(shù)據(jù)變更就能自動檢測到的,執(zhí)行變更檢測的一些情況有:組件中的輸入發(fā)生變化、組件中有事件響應、setTimeOut函數(shù)等。

這樣在上面的小例子中, @ViewChild(TodoComponent)todoComponent: TodoComponent;從undefined到[object Object],而并沒有觸發(fā)ng的變更檢測。

解決辦法也很簡單,ng支持手動觸發(fā)變更檢測,只要在適當?shù)奈恢?,調用變更檢測即可。
在上面的例子中,解決辦法為:

從@angular/core引入AfterViewInit, ChangeDetectorRef。注入ChangeDetectorRef對象,并在聲明周期鉤子ngAfterViewInit中調用變更

constructor(private todoService:TodoService, private cdr: ChangeDetectorRef){}

ngAfterViewInit(){
 this.cdr.detectChanges();
}

ChangeDetectorRef

用來處理ng變更的類,可以使用它來進行完全的手動變更檢測,主要有一下方法:

1.markForCheck()標記為需要進行變更檢測,官方給的一下例子,setInterval不會觸發(fā)變更檢測,因此模板上的numberOfTicks 并不會發(fā)生變化。

setInterval(() => {
this.numberOfTicks ++
 // the following is required, otherwise the view will not be updated
 this.ref.markForCheck();
}, 1000);

2.detach()從變更檢測樹上分離,即該組件不會進行自動的變更檢測,變更需要手動進行,使用detectChanges函數(shù)。

3.detectChanges()手動檢測變更,當變更檢測代價較大時,可以設置為定時進行表更檢測

ref.detach();
setInterval(() => {
 this.ref.detectChanges();
}, 5000);

4.checkNoChanges()進行變更檢測,有變更時拋出異常

5.reattach()detach()方法的作用相反

其他一些變更檢測知識

angular2中的每一個組件都關聯(lián)到一個變更檢測器,ChangeDetectorRef可以用來控制變更檢測器進行檢測。
瀏覽器的以下行為可以出發(fā)檢測器進行檢測:

1.所有瀏覽器事件

2.setTimeout()setInterval()

3.Ajax請求

OnPush變更檢測模式

組件默認使用的是Default變更檢測模式,只要組件的輸入發(fā)生變化時,就會觸發(fā)檢測器的執(zhí)行。除Default模式外,還有一種OnPush變更檢測模式,使用該模式首先需要在組件聲明修飾符中添加

@Component({
 selector: 'todo-list',
 changeDetection: ChangeDetectionStrategy.OnPush,
})

聲明為OnPush變更檢測模式意味著當組件輸入發(fā)生變化時,不一定會觸發(fā)變更檢測器,只有當該輸入的引用發(fā)生變化時,檢測器才會觸發(fā)。例如在一個數(shù)組中某個下標的值發(fā)生變化時,檢測器不會觸發(fā),視圖不會更新,只有該數(shù)組引用發(fā)生變化時,視圖才會更新。當然瀏覽器事件、observable發(fā)出的事件等還是會觸發(fā)檢測器的。

總結

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家腳本之家的支持。

您可能感興趣的文章:

相關文章

  • angular共享依賴的解決方案分享

    angular共享依賴的解決方案分享

    這篇文章主要給大家介紹了關于angular共享依賴解決方案的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-10-10
  • Angular中Router的常用類分享

    Angular中Router的常用類分享

    Angular Router模塊提供了幾個重要的類來處理路由信息,這篇文章小編就來和大家分享一下這幾個類的使用吧,需要的小伙伴可以收藏一下
    2023-06-06
  • angularJS開發(fā)注意事項

    angularJS開發(fā)注意事項

    本篇文章給大家分享了angularJS開發(fā)注意事項以及相關知識點,對此有興趣的朋友參考學習下。
    2018-05-05
  • 如何在AngularJs中調用第三方插件庫

    如何在AngularJs中調用第三方插件庫

    在AngularJs中我們會不可避免的使用第三方庫,這篇文章主要介紹了如何在AngularJs中調用第三方插件庫,有興趣的可以了解下
    2017-05-05
  • AngularJS自定義指令實現(xiàn)面包屑功能完整實例

    AngularJS自定義指令實現(xiàn)面包屑功能完整實例

    這篇文章主要介紹了AngularJS自定義指令實現(xiàn)面包屑功能,結合完整實例形式分析了AngularJS自定義指令的定義、調用及面包屑功能的具體實現(xiàn)技巧,需要的朋友可以參考下
    2017-05-05
  • AngularJS使用ng-app自動加載bootstrap框架問題分析

    AngularJS使用ng-app自動加載bootstrap框架問題分析

    這篇文章主要介紹了AngularJS使用ng-app自動加載bootstrap框架問題,分析了前面文章中所述的ng-app自動加載bootstrap出現(xiàn)的錯誤原因與相應的解決方法,需要的朋友可以參考下
    2017-01-01
  • angularjs使用directive實現(xiàn)分頁組件的示例

    angularjs使用directive實現(xiàn)分頁組件的示例

    本篇文章主要介紹了angularjs使用directive實現(xiàn)分頁組件的示例,具有一定的參考價值,有興趣的可以了解一下。
    2017-02-02
  • Angular模版驅動表單的使用總結

    Angular模版驅動表單的使用總結

    這篇文章主要介紹了Angular模版驅動表單的使用總結,本文實現(xiàn)了Angular支持表單的雙向數(shù)據(jù)綁定,校驗,狀態(tài)管理,非常具有實用價值,需要的朋友可以參考下
    2018-05-05
  • Angular頁面間切換及傳值的4種方法

    Angular頁面間切換及傳值的4種方法

    這篇文章主要為大家詳細介紹了Angular頁面間切換及傳值的四種方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-11-11
  • AngularJS $injector 依賴注入詳解

    AngularJS $injector 依賴注入詳解

    這篇文章主要介紹了AngularJS $injector 依賴注入的相關資料,需要的朋友可以參考下
    2016-09-09

最新評論