详解Angular组件之生命周期(二)

view钩子有2个,ngAfterViewInit和ngAfterViewChecked钩子。

1、实现ngAfterViewInit和ngAfterViewChecked钩子时注意事项

以父组件调用子组件方法中例子为基础,在父组件中实现ngAfterViewInit和ngAfterViewChecked钩子。

这两个钩子是在组件的模版所有内容组装完成后,组件模版已经呈现给用户看了,之后这两个钩子方法会被调用。

@ViewChild('child1') child1:Child1Component; //父组件中获得子组件的引用 ngOnInit(){ this.child1.greeting("Tom"); } ngAfterViewInit(){ console.log("父组件的视图初始化完毕"); } ngAfterViewChecked(){ console.log("父组件的视图变更检测完毕"); }

在子组件中也实现这两个钩子

export class Child1Component implements OnInit,AfterViewInit,AfterViewChecked{ constructor() { } ngOnInit() { } greeting(name: string) { console.log("hello" + name); } ngAfterViewInit(){ console.log("子组件的视图初始化完毕"); } ngAfterViewChecked(){ console.log("子组件的视图变更检测完毕"); } }

在父组件的ngOnInit中不直接调用子组件的greeting()方法,而是通过一个定时器每隔5s去调用一次。

ngOnInit(){ setInterval(()=>{ this.child1.greeting("Tom"); },5000); }

详解Angular组件之生命周期(二)

总结:

1、Init先调用,checked后调用

看1中,首先子组件视图初始化完毕,然后子组件视图变更检测完毕。

2、子组件先于父组件被组装好

看2中,因为父组件中声明了2个子组件,所以看到有2个子组件 初始化的动作。1号子组件初始化完毕,变更检测完毕,2号子组件初始化完毕,变更检测完毕后,父组件的初始化完毕才会被调用,然后父组件的变更检测完毕才会被调用。

3、ngAfterViewInit只会在初始化完毕被调用一次。

4、定时器触发方法后,两个子组件的变更检测会被调用,父组件的变更检测也会被调用。

视图没有发生任何改变,变更检测也会被调用,实现来ngAfterViewChecked()钩子的方法都会被调用。

所以ngAfterViewChecked()钩子一定要写的精简以免出现性能问题。

2、在一个变更检测周期中禁止一个视图被组装好之后再去更新视图

例子:

父组件

有一个message初始化为abc.显示到模版上。

message:string='abc';

在父组件的ngAfterViewInit中更改message值。

ngAfterViewInit(){ console.log("父组件的视图初始化完毕"); this.message="def"; }

详解Angular组件之生命周期(二)

会报错。ngAfterViewInit()和ngAfterViewChecked()都是在视图组装完成后触发的,所以在这两个钩子中更新组件中被绑定的属性,触发组件视图的变化,Angular就会抛出异常。

解决办法:

把代码放在另一个时间循环里面。

ngAfterViewInit(){ console.log("父组件的视图初始化完毕"); setTimeout(()=>{ this.message="def"; },0); }

二、content钩子

包括2个与投影相关的钩子,ngAfterContentInit()和ngAfterContentChecked()钩子。

ngAfterContentInit,ngAfterContentChecked和ngAfterViewInit,ngAfterViewChecked类似。

ngAfterViewInit,ngAfteViewChecked是在整个组件的视图全部组装完成后调用的。

ngAfterContentInit,ngAfterContentChecked是在被投影进来的内容组装完成后调用的。

1、Content钩子的调用顺序例子

父组件中实现ngAfterContentInit,ngAfterContentChecked和ngAfterContentInit()

export class AppComponent implements OnInit, AfterViewInit, AfterContentInit,AfterContentChecked{ ngAfterViewInit(){ console.log("父组件的视图初始化完毕"); } ngAfterContentInit(){ console.log("父组件投影内容初始化完毕"); } ngAfterContentChecked(){ console.log("父组件投影内容变更检测完毕"); }

子组件中也实现这3个接口

export class Child2Component implements OnInit,AfterViewChecked,AfterContentInit,AfterContentChecked{ constructor() { } ngOnInit() { } ngAfterViewInit(){ console.log("子组件的视图初始化完毕"); } ngAfterContentInit(){ console.log("子组件投影内容初始化完毕"); } ngAfterContentChecked(){ console.log("子组件投影内容变更检测完毕"); } }

详解Angular组件之生命周期(二)

组装视图时,先组装投影进来的内容,然后组装子组件中视图的内容,再加上父组件本身的内容,然后才是父组件视图初始化完毕。

2、Content钩子中可以修改模版内容

view钩子里不能修改模版内容,因为模版内容组装完毕后不能再修改里面内容。但是Content钩子里可以。

因为Content钩子调用时整个视图还没有组装完毕,只是投影进来的内容被组装完毕了。

父组件中在ngAfterContentInit钩子里修改message信息不会报错。

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/zygzzs.html