最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Why is ngOnChange not detecting @Input element change whil ngOnDetect was able to do so - Stack Overflow

programmeradmin1浏览0评论

Consider this plunker

Note: To observe the effect you have to rerun the app after the entering the link

import {Component, OnInit, Input, OnChanges, DoCheck} from 'angular2/core'

@Component({
  selector: 'sub',
  template: `
    <li  *ngFor="#elem of sub_list">
      <div>{{elem['name']}}</div>
    </li>
    `
})
export class Sub {

  @Input()
  sub_list: Object[];

  ngOnInit(){
    console.log('init');
    console.log(this.sub_list);
  } 

  ngOnChanges(){
    console.log('change');
    console.log(this.sub_list);
  }

  ngDoCheck() {
    console.log('check');
    console.log(this.sub_list);
  }

}
@Component({
  selector: 'my-app',
  template: `
    <div>
      <sub
        [sub_list]="my_list"
      >
      </sub>

    </div>

    `,
  directives: [Sub]
})
export class App {

  my_list: Object[] = [];

  ngOnInit() {
      var vm = this;

    setTimeout(function() {
          for(var i = 0; i < 100; i++) {

        vm.my_list.push({
          'index' : i,
          'name'  : i
        });
      }
    }, 100);

  }
}

If I try to print out this.sub_list in Sub's ngOnChange the browser outputs an empty list.

However we can see that ngDoCheck still correctly captures the change.

Is there a specific reason for this?

Consider this plunker

Note: To observe the effect you have to rerun the app after the entering the link

import {Component, OnInit, Input, OnChanges, DoCheck} from 'angular2/core'

@Component({
  selector: 'sub',
  template: `
    <li  *ngFor="#elem of sub_list">
      <div>{{elem['name']}}</div>
    </li>
    `
})
export class Sub {

  @Input()
  sub_list: Object[];

  ngOnInit(){
    console.log('init');
    console.log(this.sub_list);
  } 

  ngOnChanges(){
    console.log('change');
    console.log(this.sub_list);
  }

  ngDoCheck() {
    console.log('check');
    console.log(this.sub_list);
  }

}
@Component({
  selector: 'my-app',
  template: `
    <div>
      <sub
        [sub_list]="my_list"
      >
      </sub>

    </div>

    `,
  directives: [Sub]
})
export class App {

  my_list: Object[] = [];

  ngOnInit() {
      var vm = this;

    setTimeout(function() {
          for(var i = 0; i < 100; i++) {

        vm.my_list.push({
          'index' : i,
          'name'  : i
        });
      }
    }, 100);

  }
}

If I try to print out this.sub_list in Sub's ngOnChange the browser outputs an empty list.

However we can see that ngDoCheck still correctly captures the change.

Is there a specific reason for this?

Share asked Apr 16, 2016 at 23:46 testingtesting 2,3434 gold badges24 silver badges33 bronze badges
Add a ment  | 

3 Answers 3

Reset to default 5

In your case ngOnChanges won't be called for updates on the array. As a matter of fact, Angular2 detects updates based on references. I mean if the reference of the whole array doesn't change (it's the case when adding an element in it using the push method), the ngOnChanges method won't be called.

In your case your array is null when ngOnChanges is called because it's called before the input element is set.

There are two ways to detect changes in this case:

  • update the whole array reference using methods like slice (after the push) or concat.

    this.myArray.push({...});
    this.myArray.push = this.myArray.push.slice();
    
  • leverage the ngDoCheck method with the IterableDiffers class to manually check updates. The class allows you to register callbacks to be notified when an element is added (or remove) in an array.

See these links for more details:

  • Angular2 change detection: ngOnChanges not firing for nested object
  • https://angular.io/docs/ts/latest/api/core/DoCheck-interface.html

Ok, i've got it. ngOnChanges is fired when you initialize class field with empty array, then you update array within timeout callback and as Thierry correctly pointed changes are not detected.

Angular includes zone.js library that can track all async events in you app. Zone detects that timeout callback was executed somewhere and triggers ngDoCheck cycle and you get correct log.

see plunkr

ngOnChanges do not detect nested Object changes when we have same reference, the solution is to create new object reference each time :

const object = Object.assign({},oldObject)

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论