浅谈对Angular中的生命周期钩子的理解


Posted in Javascript onJuly 31, 2017

本文介绍了Angular中的生命周期钩子的理解,分享给大家,希望对大家有所帮助

什么是生命周期钩子

简单点来说生命周期钩子就是Angular中一个组件从被创建当销毁期间的一些有意义的关键时刻.这些关键时刻在Angular中被Angular核心模块 @angular/core 暴露出来,赋予了我们在它们发生时采取行动的能力.

有哪些生命周期钩子

Angular中从一个组件的创建到销毁一个有八个生命周期钩子它们,按照先后顺序.它们分别是:

  1. ngOnChanges()
  2. ngOnInit()
  3. ngDoCheck()
  4. ngAfterContentInit()
  5. ngAfterContentChecked()
  6. ngAfterViewInit()
  7. ngAfterViewChecked()
  8. ngOnDestroy()

其中: ngOnInit() 、 ngAfterContentInit() 、 ngAfterViewInit() 和 ngOnDestroy() 在一个组件的生命周期中只会被调用一次,其它的都有可能会被多次调用.下面,就让我们来详细解列一下这些生命周期钩子.

ngOnChanges()

  • 当Angular(重新)设置数据绑定输入属性时响应。 该方法接受当前和上一属性值的SimpleChanges对象
  • 当被绑定的输入属性的值发生变化时调用,首次调用一定会发生在ngOnInit()之前。

ngOnChanges() 生命周期的调用与一个组件中的输入属性有关.

当在一个组件中使用 @Input() 定义了一个输入属性时.只要这个输入属性的值发生了改变.就会触发 ngOnChanges() 生命周期钩子.这个生命周期钩子被调用时会传入一个 SimpleChanges 对象,这个对象中包含了输入属性当前值和上一值.

@Input()
public name: string;

ngOnChanges(changes: SimpleChanges): void {
 console.log(changes); // name:SimpleChange {previousValue: "a", currentValue: "ab", firstChange: false}
}

上面是我定义了一个输入属性 name 并将从 a 它改为 ab 之后的打印结果,可能你还注意到了打印的结果中还有一个 firstChange 属性.它是一个Booleans,表明你是否是第一次改变.

同时,还有一点需要注意:你的输入属性定义为你引用类型和基本类型的时候其表现结果是不同的.当你的输入属性是基本类型时.你的每一次改变都会触发 ngOnChanges() 生命周期钩子,而当你的输入属性是引用类型时,你改变你引用类型 当中 的属性时,并不会触发 ngOnChanges() 生命周期钩子.只有当你将你引用类型数据的指针指向另一块内存地址的时候才会触发 ngOnChanges() 生命周期钩子.

ngOnInit()

  • 在Angular第一次显示数据绑定和设置指令/组件的输入属性之后,初始化指令/组件。
  • 在第一轮ngOnChanges()完成之后调用,只调用一次。

ngOnInit() 是一个组件的生命周期中一定存在的一个钩子.它在一个组件被初始化的时候被调用.在这个期间,你可以执行一些相应的数据绑定操作.

ngDoCheck()

  • 检测,并在发生Angular无法或不愿意自己检测的变化时作出反应。
  • 在每个Angular变更检测周期中调用,ngOnChanges()和ngOnInit()之后。

ngDoCheck() 是Angular中的变更检测机制.它由 zone.js 来实现的.其行为是只要你的Angular中的某个组件发生异步事件.就会检查整个组件树,以保证组件属性的变化或页面的变化是同步的.所以 ngDoCheck() 的触发相当频繁的.并且是我们无法预料到的.也许我们在页面上的一个无意识操作,就会触发几个甚至几十个的 ngDoCheck() 生命周期钩子.

所以我们在使用 ngDoCheck() 生命周期钩子的时候一定要加上判断.以避免无用的触发干扰我们.

ngAfterContentInit()

  • 当把内容投影进组件之后调用。
  • 第一次ngDoCheck()之后调用,只调用一次。
  • 只适用于组件。

当父组件向子组件投影内容的时.在子组件内会初始化父组件的投影内容,此时会调用 ngAfterContentInit() 生命周期钩子.在整个组件生命周期中 ngAfterContentInit() 生命周期钩子只会调用一次.如下所示:

// 父组件
<app-child>
 <p>我是父组件向子组件的投影内容</>
</app-child>


// 子组件 ChildComponent
<div>
 //接受父组件的投影内容
 <ng-content></ng-content>
</div>

ngAfterContentChecked()

  • 每次完成被投影组件内容的变更检测之后调用。
  • ngAfterContentInit()和每次ngDoCheck()之后调用
  • 只适合组件。

当父组件向子组件的投影内容发生改变时会调用 ngAfterContentChecked() 生命周期钩子.它与 ngDoCheck() 类似.当投影内容发生改变时,就会执行变更检查机制.同时调用 ngAfterContentChecked() 生命周期钩子.此外.还有一点:当父组件和子组件都有投影内容时,会先执行父组件的生命周期钩子.它与下面要说的 ngAfterViewInit() 和 ngAfterViewChecked() 相反.

ngAfterViewInit()

  • 初始化完组件视图及其子视图之后调用。
  • 第一次ngAfterContentChecked()之后调用,只调用一次。
  • 只适合组件。

当其组件本身和所有的子组件渲染完成,已经呈现在页面上时,调用 ngAfterViewInit() 生命周期钩子.在整个组件生命周期中 ngAfterViewInit() 生命周期钩子只会调用一次.

ngAfterViewChecked()

  • 每次做完组件视图和子视图的变更检测之后调用。
  • ngAfterViewInit()和每次ngAfterContentChecked()之后调用。
  • 只适合组件。

当组件及其子组件的视图发生改变时,执行完变更检查机制后调用.当父组件和子组件都发生视图变化时,会先执行子组件的生命周期钩子.

注意:这里所说的视图发生改变不一定是真正页面上的变化.只是Angular种所认为的视图变化.因为Angular本身并不能察觉到页面上显示的视图.所以在Angular认为,只要你在后台定义的属性发生了改变,就是视图有了变化.从而就会调用 ngAfterViewChecked() 生命周期钩子.

ngOnDestroy

  • 当Angular每次销毁指令/组件之前调用并清扫。 在这儿反订阅可观察对象和分离事件处理器,以防内存泄漏。
  • 在Angular销毁指令/组件之前调用。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
解决jquery submit()提交表单提示:f[s] is not a function
Jan 23 Javascript
JavaScript获取onclick、onchange等事件值的代码
Jul 22 Javascript
js中return false(阻止)的用法
Aug 14 Javascript
在JavaScript中处理数组之reverse()方法的使用
Jun 09 Javascript
javascript实现网页屏蔽Backspace事件,输入框不屏蔽
Jul 21 Javascript
基于HTML模板和JSON数据的JavaScript交互(移动端)
Apr 06 Javascript
angular ng-click防止重复提交实例
Jun 16 Javascript
Javascript实现跨域后台设置拦截的方法详解
Aug 04 Javascript
JS开发中基本数据类型具体有哪几种
Oct 19 Javascript
在vue中v-bind使用三目运算符绑定class的实例
Sep 29 Javascript
用js限制网页只在微信浏览器中打开(或者只能手机端访问)
Dec 24 Javascript
JavaScript实现省份城市的三级联动
Feb 11 Javascript
Bootstrap Table 在指定列中添加下拉框控件并获取所选值
Jul 31 #Javascript
vue组件 $children,$refs,$parent的使用详解
Jul 31 #Javascript
Kindeditor单独调用单图上传增加预览功能的实例
Jul 31 #Javascript
Kindeditor单独调用多图上传实例
Jul 31 #Javascript
老生常谈ES6中的类
Jul 31 #Javascript
JS时间控制实现动态效果的实例讲解
Jul 31 #Javascript
JavaScript实现简单的双色球(实例讲解)
Jul 31 #Javascript
You might like
根据ip调用新浪api获取城市名并转成拼音
2014/03/07 PHP
PHP+Mysql基于事务处理实现转账功能的方法
2015/07/08 PHP
XP折叠菜单&amp;仿QQ2006菜单
2006/12/16 Javascript
javascript 读取图片文件的大小
2009/06/25 Javascript
javascript 全选与全取消功能的实现代码
2012/12/23 Javascript
全面解析JavaScript中的valueOf与toString方法(推荐)
2016/06/14 Javascript
深入理解bootstrap框架之第二章整体架构
2016/10/09 Javascript
Angular 2父子组件数据传递之@Input和@Output详解 (上)
2017/07/05 Javascript
js 事件的传播机制(实例讲解)
2017/07/20 Javascript
jQuery中.attr()和.data()的区别分析
2017/09/03 jQuery
bootstrap日期插件daterangepicker使用详解
2017/10/19 Javascript
JavaScript的setter与getter方法
2017/11/29 Javascript
jquery如何实现点击空白处隐藏元素
2017/12/05 jQuery
JS与jQuery判断文本框还剩多少字符可以输入的方法
2018/09/01 jQuery
vue的全局变量和全局拦截请求器的示例代码
2018/09/13 Javascript
JS实现点击下拉列表文本框中出现对应的网址,点击跳转按钮实现跳转
2019/11/25 Javascript
Node.js API详解之 readline模块用法详解
2020/05/22 Javascript
Python 异常处理实例详解
2014/03/12 Python
bat和python批量重命名文件的实现代码
2016/05/19 Python
python实现下载文件的三种方法
2017/02/09 Python
Python实现一个Git日志统计分析的小工具
2017/12/14 Python
用pandas中的DataFrame时选取行或列的方法
2018/07/11 Python
python3实现网络爬虫之BeautifulSoup使用详解
2018/12/19 Python
python中Lambda表达式详解
2019/11/20 Python
python 实现兔子生兔子示例
2019/11/21 Python
解决Tensorboard可视化错误:不显示数据 No scalar data was found
2020/02/15 Python
CSS 说明横向进度条最后显示文字的实现代码
2020/11/10 HTML / CSS
JOSEPH官网:英国奢侈时尚品牌
2018/01/31 全球购物
雅诗兰黛澳大利亚官网:Estée Lauder澳大利亚
2019/05/31 全球购物
运行时异常与一般异常有何异同?
2014/01/05 面试题
学习经验交流会主持词
2014/04/01 职场文书
舞蹈教育学专业自荐信
2014/06/15 职场文书
课前一分钟演讲稿
2014/08/26 职场文书
党员教师群众路线对照检查材料思想汇报
2014/09/29 职场文书
SQL Server中使用判断语句(IF ELSE/CASE WHEN )案例
2021/07/07 SQL Server
Python实现聚类K-means算法详解
2022/07/15 Python