angular2系列教程(五)Structural directives、再谈组件生命周期

今天,我们要讲的是structural directives和组件生命周期这两个知识点。structural directives顾名思义就是改变dom结构的指令。著名的内建结构指令有 ngIfngSwitch and ngFor

例子

例子是我自己改写的,编写一个structural directives,然后通过这个指令实例化和注销组件,在此同时监视组件生命周期。

源代码

UnlessDirective

这个指令是官网示例中的指令。

src/unless.directive.ts

import {Directive, Input} from ‘angular2/core‘;
import {TemplateRef, ViewContainerRef} from ‘angular2/core‘;
@Directive({ selector: ‘[myUnless]‘ })
export class UnlessDirective {
  constructor(
    private _templateRef: TemplateRef,
    private _viewContainer: ViewContainerRef
    ) { }
  @Input() set myUnless(condition: boolean) {
    if (!condition) {
      this._viewContainer.createEmbeddedView(this._templateRef);
    } else {
      this._viewContainer.clear();
    }
  }
}

通过注入TemplateRef, ViewContainerRef这两个服务,来控制template的实例化和注销。TemplateRef可以让我们获取指令所在的元素的template,ViewContainerRef提供了多种视图容器的方法。

更详细的介绍:

用于测试的组件

接下来我们编写一个用于测试的组件。

src/lifecycle.ts

import {Component,Input} from ‘angular2/core‘;
import {bootstrap} from ‘angular2/platform/browser‘;
import {OnChanges, SimpleChange,OnInit,AfterContentInit,AfterContentChecked,AfterViewInit,AfterViewChecked,OnDestroy} from ‘angular2/core‘;

@Component({
    selector: "lifecycle",
    template: `
    <div>
    <span>{{name}}</span>
     <button (click)="doSomething()">click and watch the lifecycle</button>
     </div>
    `
})
export class Lifecycle
implements OnChanges, OnInit,AfterContentInit,AfterContentChecked,AfterViewInit, AfterViewChecked, OnDestroy{
    @Input()
    name:string
    doSomething(){
        console.log(‘***********doSomething**********‘);
        setTimeout(()=>{
             console.log(‘***********setTimeout**********‘);
            this.name=‘susan‘
        },1000)
    }
    ngOnInit(){console.log(‘onInit‘);}
    ngOnDestroy(){console.log(‘OnDestroy‘)}
    ngOnChanges(changes: {[propertyName: string]: SimpleChange}){console.log(‘ngOnChanges‘,changes)}
    ngAfterContentInit(){console.log(‘AfterContentInit‘)}
    ngAfterContentChecked(){console.log(‘AfterContentChecked‘)}
    ngAfterViewInit(){console.log(‘AfterViewInit‘)}
    ngAfterViewChecked(){console.log(‘AfterViewChecked‘)}
}

这段代码我们做了这些事:

  1. 渲染一个span一个button
  2. 设置成员变量name,@input代表从parent输入
  3. 设置成员函数doSomething,打印一个信息,执行一个异步操作setTimeout
  4. 继承接口,设置所有的生命周期钩子,并打印信息

我们将使用这个组件,来监视组件生命周期。

使用指令控制组件

我们将我们的组件渲染出来,并用我们编写的结构指令“myunless”去实例化和注销这个组件

src/app.ts

import {Component} from ‘angular2/core‘;
import {bootstrap} from ‘angular2/platform/browser‘;
import {UnlessDirective}from ‘./unless.directive‘;
import {Lifecycle} from ‘./lifecycle‘

@Component({
    selector: "app",
    directives:[UnlessDirective,Lifecycle],
    template: `
      <button
      (click)="condition = !condition"
      [style.background] = "condition ? ‘orangered‘: ‘lightgreen‘"
      >
      Set ‘condition‘ to {{condition ? ‘False‘: ‘True‘}}
      </button>

      <lifecycle *myUnless="condition" name="lewis"></lifecycle>
    `
})
export class App {
    constructor() {}
}

bootstrap(App, [])
    .catch(err => console.error(err));    

这段代码我们干了这些事:

  1. 注入组件和指令directives:[UnlessDirective,Lifecycle]
  2. 渲染一个button控制成员变量condition的正负
  3. 渲染我们的组件lifecycle,并使用指令控制它的实例化和注销<lifecycle *myUnless="condition" name="lewis"></lifecycle>
  4. 最后启动这个app组件bootstrap(App, []) .catch(err => console.error(err));

开始测试

刷新页面:

执行了一次Onchanges、onInit、AfterContentInit、AfterViewInit,执行了五次AfterContentChecked和AfterViewChecked,

  1. onInit是在组件第一次ngOnChanges时执行
  2. OnChanges在input和output绑定的值变化时候;我们可以看到打印了变化的值。可以替代ng1中的$watch;
  3. AfterContentInit、AfterViewInit分别代表在组件内容和视图初始化后执行。
  4. AfterContentChecked和AfterViewChecked是在组件内容和视图检查完后执行。

这里没有DoCheck,因为接口没有证实。

点击Set ‘condition‘ toTrue按钮,页面上的组件被注销

console打印:

点击Set ‘condition‘ to False按钮,页面上的组件重新被实例化:

console打印:

打印了一次Onchanges、onInit、AfterContentInit、AfterViewInit、AfterContentChecked和AfterViewChecked,说明组件实例化,只需要触发一轮初始化和变化检查。与刷新页面的五次对比,我们可以知道多余的“变化检查”,可能来源于angualr的启动。

点击click and watch the lifecycle按钮,一秒后页面上的name变为susan:

console打印

先打印一次AfterContentChecked和AfterViewChecked,一秒后又打印两次。OnChanges没有触发。

结论和收获

  1. TemplateRef, ViewContainerRef这两个服务可以帮助我们实现结构指令的编写
  2. 结构指令可以完全注销组件,节约性能消耗
  3. 组件实例化,只需要触发一轮初始化和“变化检查”
  4. angualr的启动会触发多次“变化检查”
  5. 我们可以继承OnChanges接口,来实现类似ng1中的$watch功能,获取变化前后的值,但是只能监视@input装饰的变量
  6. ng2使用zone,将window对象上常见的异步方法(setTimeout等),都打上了“猴子补丁”,使其可以直接更新视图,我们再也不用在异步中写ng1中的$apply了
  7. 我们可以使用setTimeout(()=>{},0),在浏览器的一轮“event loop”后来触发ng2的“变化检查”
  8. 我们触发类的成员函数(doSomething)时,也会导致ng2的“变化检查”

教程源代码及目录

如果您觉得本博客教程帮到了您,就赏颗星吧!

https://github.com/lewis617/angular2-tutorial

时间: 2024-10-18 04:40:52

angular2系列教程(五)Structural directives、再谈组件生命周期的相关文章

再谈Activity生命周期

首先来幅很经典的图,看完之后再说话 1.启动一个新的Activity A,执行onCreate() -> onStart() -> onResume,此时该Activity获取焦点并且运行在前台. 2.当Activity A失去焦点但是仍然可见,也就是当我们重新启动一个新的Activity B,Activity B是透明的或者尺寸并不是全屏显示,这样我们还是可以看到底部的Activity A,但是它被Activity B覆盖或者部分覆盖,所以此时Activity A会执行onPause()方法

C#微信公众号开发系列教程五(接收事件推送与消息排重)

微信公众号开发系列教程一(调试环境部署) 微信公众号开发系列教程一(调试环境部署续:vs远程调试) C#微信公众号开发系列教程二(新手接入指南) C#微信公众号开发系列教程三(消息体签名及加解密) C#微信公众号开发系列教程四(接收普通消息) C#微信公众号开发系列教程五(接收事件推送与消息排重) 在上一篇的博文中讲到,微信的消息可以大体分为两种类型,一种是包括:文本,语音,图片等的普通消息,另一种就是本篇要将的事件类型.包括:关注/取消关注事件,扫描带参数二维码事件,上报地理位置事件,自定义菜

angular2系列教程(十)两种启动方法、两个路由服务、引用类型和单例模式的妙用

今天我们要讲的是ng2的路由系统. 例子 例子是官网的例子,包含一个“危机中心”和“英雄列表”,都在一个app中,通过路由来控制切换视图.还包含了promise的用法,服务的用法等多个知识点. 源代码: https://github.com/lewis617/angular2-tutorial/tree/gh-pages/router 运行方法: 在根目录下运行: http-server 引入库文件设置base href 路由并不在ng2中,需要我们额外引入,另外我们需要设置base href,

angular2系列教程(六)升级装备、pipe

今天,我们要讲的是angualr2的pipe这个知识点,但是在这之前我们需要升级一下我们的装备,因为之前的装备太“寒酸”了. 例子 这个例子包含两个pipe,一个是stateful,一个是stateless,是直接复制官方的例子.本例子还包含了我对AngularClass/angular2-webpack-starter这个牛逼starter的改写,我会详细讲解配置. 源代码 没有测试 AngularClass/angular2-webpack-starter 这里面包含了Angular 2 (

angular2系列教程(一)hello world

今天我们要讲的是angular2系列教程的第一篇,主要是学习angular2的运行,以及感受conponents以及模板语法. http://lewis617.github.io/angular2-tutorial/hellowold/ 例子 这个例子非常简单,是个双向数据绑定. 运行方法: 全局安装http-server npm install -g http-server 公共部分 公共部分就是你直接复制粘贴拿去用的部分,包括: 1.index.html 2.lib 3.app/main.t

MyBatis系列教程(五)-- 处理存储过程(Handle the Store Procedure)

所需要用到的其他工具或技术: 项目管理工具 : Maven 测试运行工具 : Junit 数据库 : Derby 本节需要用到的有2部分,第一部分是如何在Derby中创建存储过程,第二部分是如何在Mybatis中调用存储过程 一. 在Derby中创建存储过程 1.在Eclipse中创建一个新的普通java项目命名为Test_Store_Procedure 2.在com.freud.practice包下创建一个Class命名为StoreProcedureOperationClass.class p

【REACT NATIVE 系列教程之二】创建自定义组件&&导入与使用示例

本站文章均为 李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/react-native/2219.html 在上一篇  [REACT NATIVE 系列教程之一]触摸事件的两种形式与四种TOUCHABLE组件详解 中的最后介绍了如何使用Touchable的四种组件进行监听触摸事件.  那么紧接着我们利用Touchable来包装一个带图片的Button组件,且设计成可接受很多自定义参数. 一:创建我们自定义

Android系列之Fragment(二)Fragment的生命周期和返回栈

Android系列之Fragment(二)Fragment的生命周期和返回栈 - Android - 次元立方网 - 电脑知识与技术互动交流平台 [正文] 上一章节中(Android系列之Fragment(一)----Fragment加载到Activity当中),我们对Fragment的生命周期进行了简单介绍,这一章节将对生命周期和返回栈进行详细介绍. 一.Fragment的生命周期初探: 因为Fragment必须嵌入在Acitivity中使用,所以Fragment的生命周期和它所在的Activ

Angular2 组件生命周期

1. 说明 Angular每个组件都存在一个生命周期,从创建,变更到销毁.Angular提供组件生命周期钩子,把这些关键时刻暴露出来,赋予在这些关键结点和组件进行交互的能力. 2. 接口 按照生命周期执行的先后顺序,Angular生命周期接口如下所示 名称 时机 接口 范围 ngOnChanges 当被绑定的输入属性的值发生变化时调用,首次调用一定会发生在 ngOnInit之前. OnChanges 指令和组件 ngOnInit 在第一轮 ngOnChanges 完成之后调用. ( 译注:也就是