Angular 是一个用 HTML 和 JavaScript 或者一个可以编译成 JavaScript 的语言(例如 Dart 或者 TypeScript ),来构建客户端应用的框架。

该框架包括一系列库,有些是核心库,有些是可选库。

我们是这样写 Angular 应用的:用 Angular 扩展语法编写 HTML 模板,用组件类管理这些模板,用服务添加应用逻辑,用模块打包发布组件与服务。

然后,我们通过引导根模块来启动该应用。Angular 在浏览器中接管、展现应用的内容,并根据我们提供的操作指令响应用户的交互。

当然,这只是冰山一角。后面我们将学习更多的细节。不过,目前我们还是先关注全景图吧。

这个架构图展现了 Angular 应用中的 8 个主要构造块:

  • 模块 (module)

  • 组件 (component)

  • 模板 (template)

  • 元数据 (metadata)

  • 数据绑定 (data binding)

  • 指令 (directive)

  • 服务 (service)

  • 依赖注入 (dependency injection)

掌握了这些构造块,你就可以下山了!

本章所引用的代码见在线例子。

The code referenced on this page is available as a live example.

模块

Angular 应用是模块化的,并且 Angular 有自己的模块系统,它被称为 Angular 模块NgModules

Angular 模块很重要。这里只是简单介绍,在 Angular 模块中会做深入讲解。



每个 Angular 应用至少有一个模块(根模块),习惯上命名为AppModule

根模块在一些小型应用中可能是唯一的模块,大多数应用会有很多特性模块,每个模块都是一个内聚的代码块专注于某个应用领域、工作流或紧密相关的功能。

Angular 模块(无论是根模块还是特性模块)都是一个带有@NgModule装饰器的类。

装饰器是用来修饰 JavaScript 类的函数。Angular 有很多装饰器,它们负责把元数据附加到类上,以了解那些类的设计意图以及它们应如何工作。关于装饰器的更多信息。

NgModule是一个装饰器函数,它接收一个用来描述模块属性的元数据对象。其中最重要的属性是:

  • declarations - 声明本模块中拥有的视图类。Angular 有三种视图类:组件、指令和管道。

  • exports - declarations 的子集,可用于其它模块的组件模板。

  • imports - 模块声明的组件模板需要的类所在的其它模块。

  • providers - 服务的创建者,并加入到全局服务列表中,可用于应用任何部分。

  • bootstrap - 指定应用的主视图(称为根组件),它是所有其它视图的宿主。只有根模块才能设置bootstrap属性。

下面是一个简单的根模块:

app/app.module.ts

import { NgModule } from '@angular/core'; import { browserModule } from '@angular/platform-browser'; @NgModule({ imports: [ browserModule ], providers: [ Logger ], declarations: [ AppComponent ], exports: [ AppComponent ], bootstrap: [ AppComponent ] }) export class AppModule { }

AppComponentexport语句只是用于演示如何导出的,它在这个例子中并不是必须的。根模块不需要导出任何东西,因为其它组件不需要导入根模块。

我们通过引导根模块来启动应用。在开发期间,你通常在一个main.ts文件中引导AppModule,就像这样:

app/main.ts

import { platformbrowserDynamic } from '@angular/platform-browser-dynamic'; import { AppModule } from './app.module'; platformbrowserDynamic().bootstrapModule(AppModule);

Angular 模块 vs. JavaScript 模块

Angular 模块(一个用@NgModule装饰的类)是 Angular 的基础特性。

JavaScript 也有自己的模块系统,用来管理一组 JavaScript 对象。它与 Angular 的模块系统完全不同且完全无关。

JavaScript 中,每个文件是一个模块,文件中定义的所有对象都从属于那个模块。通过export关键字,模块可以把它的某些对象声明为公共的。其它 JavaScript 模块可以使用import 语句来访问这些公共对象。

import { NgModule } from '@angular/core'; import { AppComponent } from './app.component';
export class AppModule { }

学习更多关于 JavaScript 模块的知识。

这两个模块化系统是互补的,我们在写程序时都会用到。

Angular 模块库

Angular 提供了一组 JavaScript 模块。可以把它们看做库模块。

每个 Angular 库的名字都带有@angular前缀。

npm 包管理工具安装它们,用 JavaScript 的import语句导入其中某些部件。



例如,象下面这样,从@angular/core库中导入Component装饰器:

import { Component } from '@angular/core';

还可以使用 JavaScript 的导入语句从 Angular 中导入 Angular 模块

import { browserModule } from '@angular/platform-browser';

在上面那个简单的根模块的例子中,应用模块需要browserModule的某些素材。要访问这些素材,就得把它加入@NgModule元数据的imports中,就像这样:

imports: [ browserModule ],

这种情况下,你同时使用了 Angular 和 JavaScript 的模块化系统。

这两个系统比较容易混淆,因为它们共享相同的词汇 “imports” 和 “exports”。先放一放,随着时间和经验的增长,自然就会澄清了。

更多信息,见 Angular 模块。

组件

组件负责控制屏幕上的一小块区域,我们称之为视图

例如,下列视图都是由组件控制的:

  • 带有导航链接的应用根组件。

  • 英雄列表。

  • 英雄编辑器。

我们在类中定义组件的应用逻辑,为视图提供支持。组件通过一些由属性和方法组成的 API 与视图交互。

例如,HeroListComponent有一个heroes属性,它返回一个英雄数组,这个数组从一个服务获得。HeroListComponent还有一个selectHero()方法,当用户从列表中点选一个英雄时,就把它/她设置到selectedHero属性。

app/hero-list.component.ts (class)

export class HeroListComponent implements OnInit { heroes: Hero[]; selectedHero: Hero; constructor(private service: HeroService) { } ngOnInit() { this.heroes = this.service.getHeroes(); } selectHero(hero: Hero) { this.selectedHero = hero; } }

当用户在这个应用中漫游时, Angular 会创建、更新和销毁组件。应用可以通过生命周期钩子在组件生命周期的各个时间点上插入自己的操作,例如上面声明的ngOnInit()

模板

我们通过组件的自带的模板来定义组件视图。模板以 HTML 形式存在,告诉 Angular 如何渲染组件。

多数情况下,模板看起来很像标准 HTML,当然也有一点不同的地方。下面是HeroListComponent组件的一个模板:

app/hero-list.component.html

   
   
<h2>Hero List</h2><p><i>Pick a hero from the list</i></p><ul> <li *ngFor="let hero of heroes" (click)="selectHero(hero)"> {{hero.name}} </li></ul><hero-detail *ngIf="selectedHero" [hero]="selectedHero"></hero-detail>

模板除了可以使用像<h2><p>这样的典型的 HTML 元素,还能使用其它元素。例如,像*ngFor{{hero.name}}(click)[hero]<hero-detail>这样的代码使用了 Angular 的模板语法。

在模板的最后一行,<hero-detail>标签就是一个用来表示新组件HeroDetailComponent的自定义元素。

HeroDetailComponent跟以前见到过的HeroListComponent不同的组件。HeroDetailComponent(代码未显示)用于展现一个特定英雄的情况,这个英雄是用户从HeroListComponent列表中选择的。HeroDetailComponentHeroListComponent子组件

注意到了吗?<hero-detail>舒适地躺在原生 HTML 元素之间。自定义组件和原生 HTML 在同一布局中融合得天衣无缝。


元数据

元数据告诉 Angular 如何处理一个类。


回头看看HeroListComponent就会明白:它只是一个类。一点框架的痕迹也没有,里面完全没有出现 "Angular" 的字样。

实际上,HeroListComponent真的只是一个类。直到我们告诉 Angular 它是一个组件。

要告诉 Angular HeroListComponent是个组件,只要把元数据附加到这个类。

在TypeScript 中,我们用装饰器 (decorator) 来附加元数据。下面就是HeroListComponent的一些元数据。

app/hero-list.component.ts (Metadata)

@Component({ moduleId: module.id, selector: 'hero-list', templateUrl: 'hero-list.component.html', providers: [ HeroService ] }) export class HeroListComponent implements OnInit { /* . . . */ }

这里看到@Component装饰器,它把紧随其后的类标记成了组件类。

@Component装饰器能接受一个配置对象, Angular 会基于这些信息创建和展示组件及其视图。

@Component的配置项包括:

  • moduleId: 为与模块相关的 URL(例如templateUrl)提供基地址。

  • selector: CSS 选择器,它告诉 Angular 在父级 HTML 中查找<hero-list>标签,创建并插入该组件。例如,如果应用的 HTML 包含<hero-list></hero-list>, Angular 就会把HeroListComponent的一个实例插入到这个标签中。

  • templateUrl:组件 HTML 模板的模块相对地址,如前所示。

  • providers - 组件所需服务的依赖注入提供商数组。这是在告诉 Angular:该组件的构造函数需要一个HeroService服务,这样组件就可以从服务中获得英雄数据。

@Component里面的元数据会告诉 Angular 从哪里获取你为组件指定的主要的构建块。

模板、元数据和组件共同描绘出这个视图。

其它元数据装饰器用类似的方式来指导 Angular 的行为。例如@Injectable@Input@Output等是一些最常用的装饰器。


这种架构处理方式是:你向代码中添加元数据,以便 Angular 知道该怎么做。

数据绑定

如果没有框架,我们就得自己把数据值推送到 HTML 控件中,并把用户的反馈转换成动作和值更新。如果手工写代码来实现这些推/拉逻辑,肯定会枯燥乏味、容易出错,读起来简直是噩梦 —— 写过 jQuery 的程序员大概都对此深有体会。

Angular 支持数据绑定,一种让模板的各部分与组件的各部分相互合作的机制。我们往模板 HTML 中添加绑定标记,来告诉 Angular 如何把二者联系起来。

如图所示,数据绑定的语法有四种形式。每种形式都有一个方向 —— 绑定到 DOM 、绑定自 DOM 以及双向绑定。


HeroListComponent示例模板中有三种形式:

app/hero-list.component.html (binding)

<li>{{hero.name}}</li> <hero-detail [hero]="selectedHero"></hero-detail> <li (click)="selectHero(hero)"></li>
  • {{hero.name}}插值表达式<li>标签中显示组件的hero.name属性的值。

  • [hero]属性绑定把父组件HeroListComponentselectedHero的值传到子组件HeroDetailComponenthero属性中。

  • (click) 事件绑定在用户点击英雄的名字时调用组件的selectHero方法。

双向数据绑定是重要的第四种绑定形式,它使用ngModel指令组合了属性绑定和事件绑定的功能。下面是HeroDetailComponent模板的范例:

app/hero-detail.component.html (ngModel)

<input [(ngModel)]="hero.name">

在双向绑定中,数据属性值通过属性绑定从组件流到输入框。用户的修改通过事件绑定流回组件,把属性值设置为最新的值。

Angular 在每个 JavaScript 事件循环中处理所有的数据绑定,它会从组件树的根部开始,递归处理全部子组件。

数据绑定在模板与对应组件的交互中扮演了重要的角色。


数据绑定在父组件与子组件的通讯中也同样重要。


指令 (directive)

Angular 模板是动态的。当 Angular 渲染它们时,它会根据指令提供的操作对 DOM 进行转换。

指令是一个带有“指令元数据”的类。在 TypeScript 中,要通过@Directive装饰器把元数据附加到类上。


组件是一个带模板的指令@Component装饰器实际上就是一个@Directive装饰器,只是扩展了一些面向模板的特性。

虽然严格来说组件就是一个指令,但是组件非常独特,并在 Angular 中位于中心地位,所以在架构概览中,我们把组件从指令中独立了出来。

还有两种其它类型的指令:结构型指令和属性 (attribute) 型指令。

它们往往像属性 (attribute) 一样出现在元素标签中,偶尔会以名字的形式出现,但多数时候还是作为赋值目标或绑定目标出现。

结构型指令通过在 DOM 中添加、移除和替换元素来修改布局。

下面的范例模板中用到了两个内置的结构型指令:

app/hero-list.component.html (structural)

<li *ngFor="let hero of heroes"></li> <hero-detail *ngIf="selectedHero"></hero-detail>
  • *ngFor告诉 Angular 为heroes列表中的每个英雄生成一个<li>标签。

  • *ngIf表示只有在选择的英雄存在时,才会包含HeroDetail组件。

属性型 指令修改一个现有元素的外观或行为。在模板中,它们看起来就像是标准的 HTML 属性,故名。

ngModel指令就是属性型指令的一个例子,它实现了双向数据绑定。ngModel修改现有元素(一般是<input>)的行为:设置其显示属性值,并响应 change 事件。

app/hero-detail.component.html (ngModel)

<input [(ngModel)]="hero.name">

Angular 还有少量指令,它们或者修改结构布局(例如 ngSwitch),或者修改 DOM 元素和组件的各个方面(例如 ngStyle和 ngClass)。

当然,我们也能编写自己的指令。像HeroListComponent这样的组件就是一种自定义指令。

服务

服务是一个广义范畴,包括:值、函数,或应用所需的特性。

几乎任何东西都可以是一个服务。典型的服务是一个类,具有专注的、明确的用途。它应该做一件特定的事情,并把它做好。


例如:

  • 日志服务

  • 数据服务

  • 消息总线

  • 税款计算器

  • 应用程序配置

服务没有什么特别属于 Angular 的特性。 Angular 对于服务也没有什么定义。它甚至都没有定义服务的基类,也没有地方注册一个服务。

即便如此,服务仍然是任何 Angular 应用的基础。组件就是最大的服务消费者。

下面是一个服务类的范例,用于把日志记录到浏览器的控制台:

app/logger.service.ts (class)

export class Logger { log(msg: any) { console.log(msg); } error(msg: any) { console.error(msg); } warn(msg: any) { console.warn(msg); } }

下面是HeroService类,用于获取英雄数据,并通过一个已解析的承诺 (Promise) 返回它们。HeroService还依赖于Logger服务和另一个用于处理服务器通讯的BackendService服务。

app/hero.service.ts (class)

export class HeroService { private heroes: Hero[] = []; constructor( private backend: BackendService, private logger: Logger) { } getHeroes() { this.backend.getAll(Hero).then( (heroes: Hero[]) => { this.logger.log(`Fetched ${heroes.length} heroes.`); this.heroes.push(...heroes); // fill cache }); return this.heroes; } }

服务无处不在。

组件类应保持精简。组件本身不从服务器获得数据、不进行验证输入,也不直接往控制台写日志。它们把这些任务委托给服务。

组件的任务就是提供用户体验,仅此而已。它介于视图(由模板渲染)和应用逻辑(通常包括模型的某些概念)之间。设计良好的组件为数据绑定提供属性和方法,把其它琐事都委托给服务。

Angular 不会强制要求我们遵循这些原则。即使我们花 3000 行代码写了一个“厨房洗碗槽”组件,它也不会抱怨什么。

Angular 帮助我们遵循这些原则 —— 它让我们能轻易地把应用逻辑拆分到服务,并通过依赖注入来在组件中使用这些服务。

依赖注入

“依赖注入”是提供类的新实例的一种方式,还负责处理好类所需的全部依赖。大多数依赖都是服务。Angular 使用依赖注入来提供新组件以及组件所需的服务。


Angular 通过查看构造函数的参数类型得知组件需要哪些服务。例如,HeroListComponent组件的构造函数需要一个HeroService服务:

app/hero-list.component.ts (constructor)

constructor(private service: HeroService) { }

当 Angular 创建组件时,会首先为组件所需的服务请求一个注入器 (injector)

注入器维护了一个服务实例的容器,存放着以前创建的实例。如果所请求的服务实例不在容器中,注入器就会创建一个服务实例,并且添加到容器中,然后把这个服务返回给 Angular。当所有请求的服务都被解析完并返回时,Angular 会以这些服务为参数去调用组件的构造函数。这就是依赖注入

HeroService注入的过程看起来有点像这样:

如果注入器还没有HeroService,它怎么知道该如何创建一个呢?

简单的说,必须在要求注入HeroService之前,在注入器中注册HeroService提供商 Provider。提供商用于创建并返回一个服务,通常是服务类本身。

我们可以在模块或组件中注册提供商。

通常会把提供商添加到根模块上,以便在任何地方使用服务的同一个实例。

app/app.module.ts (module providers)

providers: [ BackendService, HeroService, Logger ],

或者,也可以在@Component元数据中的providers属性中把它注册在组件层:

app/hero-list.component.ts (component providers)

@Component({ moduleId: module.id, providers: [ HeroService ] })

把它注册在组件级表示该组件的每一个新实例都会有一个服务的新实例。

需要记住的关于依赖注入的要点是:

  • 依赖注入渗透在整个 Angular 框架中,被到处使用。

  • 注入器 (injector) 是本机制的核心。

    • 注入器负责维护一个容器,用于存放它创建过的服务实例。

    • 注入器能使用提供商创建一个新的服务实例。

  • 提供商是一个用于创建服务的配方。

  • 提供商注册到注入器。

总结

我们学到的这些只是关于 Angular 应用程序的八个主要构造块的基础知识:

  • 模块

  • 组件

  • 模板

  • 元数据

  • 数据绑定

  • 指令

  • 服务

  • 依赖注入

这是 Angular 应用程序中所有其它东西的基础,要使用 Angular,以这些作为开端就绰绰有余了。但它仍然没有包含我们需要知道的全部。

这里是一个简短的、按字母排序的列表,列出了其它重要的 Angular 特性和服务。它们大多数已经(或即将)包括在这份开发文档中:

动画:用 Angular 的动画库让组件动起来,而不需要对动画技术或 CSS 有深入的了解。

变更检测:变更检测文档会告诉你 Angular 是如何决定组件的属性值变化,什么时候该更新到屏幕,以及它是如何利用区域 (zone) 来拦截异步活动并执行变更检测策略。

事件:事件文档会告诉你如何使用组件和服务触发支持发布和订阅的事件。

表单:通过基于 HTML 的验证和脏检查机制支持复杂的数据输入场景。

HTTP:通过 HTTP 客户端,可以与服务器通讯,以获得数据、保存数据和触发服务端动作。

生命周期钩子:通过实现生命周期钩子接口,可以切入组件生命中的几个关键点:从创建到销毁。

管道:在模板中使用管道转换成用于显示的值,以增强用户体验。例如,currency管道表达式:

price | currency:'USD':true

它把价格"42.33"显示为$42.33

路由器:在应用程序客户端的页面间导航,并且不离开浏览器。

测试:使用 Angular 测试平台,在你的应用部件与 Angular 框架交互时进行单元测试。

Angular架构的更多相关文章

  1. HTML实现代码雨源码及效果示例

    这篇文章主要介绍了HTML实现代码雨源码及效果示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. HTML文本属性&amp;颜色控制属性的实现

    这篇文章主要介绍了HTML文本属性&颜色控制属性的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  3. 简洁自适应404页面HTML好看的404源码

    这篇文章主要介绍了简洁自适应404页面HTML好看的404源码,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  4. HTML5适合的情人节礼物有纪念日期功能

    这篇文章主要介绍了HTML5适合的情人节礼物有纪念日期功能,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  5. 如何给HTML标签中的文本设置修饰线

    这篇文章主要介绍了如何给HTML标签中的文本设置修饰线,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  6. HTML5调用手机发短信和打电话功能

    这篇文章主要介绍了HTML5调用手机发短信和打电话功能,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  7. HTML利用九宫格原理进行网页布局

    这篇文章主要介绍了HTML利用九宫格原理进行网页布局,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  8. HTML中实现音乐或视频自动播放案例详解

    由于期末大作业我想插入一个背景音乐,实现点开网页就会自动播放音频的效果,今天通过本文给大家分享下我基于HTML实现音乐或视频自动播放功能,代码简单易懂,需要的朋友参考下吧

  9. 使用HTML5加载音频和视频的实现代码

    这篇文章主要介绍了使用HTML5加载音频和视频的实现代码,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  10. html svg生成环形进度条的实现方法

    这篇文章主要介绍了html svg生成环形进度条的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

随机推荐

  1. Angular2 innerHtml删除样式

    我正在使用innerHtml并在我的cms中设置html,响应似乎没问题,如果我这样打印:{{poi.content}}它给了我正确的内容:``但是当我使用[innerHtml]=“poi.content”时,它会给我这个html:当我使用[innerHtml]时,有谁知道为什么它会剥离我的样式Angular2清理动态添加的HTML,样式,……

  2. 为Angular根组件/模块指定@Input()参数

    我有3个根组件,由根AppModule引导.你如何为其中一个组件指定@input()参数?也不由AppModalComponent获取:它是未定义的.据我所知,你不能将@input()传递给bootstraped组件.但您可以使用其他方法来做到这一点–将值作为属性传递.index.html:app.component.ts:

  3. angular-ui-bootstrap – 如何为angular ui-bootstrap tabs指令指定href参数

    我正在使用角度ui-bootstrap库,但我不知道如何为每个选项卡指定自定义href.在角度ui-bootstrap文档中,指定了一个可选参数select(),但我不知道如何使用它来自定义每个选项卡的链接另一种重新定义问题的方法是如何使用带有角度ui-bootstrap选项卡的路由我希望现在还不算太晚,但我今天遇到了同样的问题.你可以通过以下方式实现:1)在控制器中定义选项卡href:2)声明一个函数来改变控制器中的散列:3)使用以下标记:我不确定这是否是最好的方法,我很乐意听取别人的意见.

  4. 离子框架 – 标签内部的ng-click不起作用

    >为什么标签标签内的按钮不起作用?>但是标签外的按钮(登陆)工作正常,为什么?>请帮我解决这个问题.我需要在点击时做出回复按钮workingdemo解决方案就是不要为物品使用标签.而只是使用divHTML

  5. Angular 2:将值传递给路由数据解析

    我正在尝试编写一个DataResolver服务,允许Angular2路由器在初始化组件之前预加载数据.解析器需要调用不同的API端点来获取适合于正在加载的路由的数据.我正在构建一个通用解析器,而不是为我的许多组件中的每个组件设置一个解析器.因此,我想在路由定义中传递指向正确端点的自定义输入.例如,考虑以下路线:app.routes.ts在第一个实例中,解析器需要调用/path/to/resourc

  6. angularjs – 解释ngModel管道,解析器,格式化程序,viewChangeListeners和$watchers的顺序

    换句话说:如果在模型更新之前触发了“ng-change”,我可以理解,但是我很难理解在更新模型之后以及在完成填充更改之前触发函数绑定属性.如果您读到这里:祝贺并感谢您的耐心等待!

  7. 角度5模板形式检测形式有效性状态的变化

    为了拥有一个可以监听其包含的表单的有效性状态的变化的组件并执行某些组件的方法,是reactiveforms的方法吗?

  8. Angular 2 CSV文件下载

    我在springboot应用程序中有我的后端,从那里我返回一个.csv文件WheniamhittingtheURLinbrowsercsvfileisgettingdownloaded.现在我试图从我的角度2应用程序中点击此URL,代码是这样的:零件:服务:我正在下载文件,但它像ActuallyitshouldbeBook.csv请指导我缺少的东西.有一种解决方法,但您需要创建一个页面上的元

  9. angularjs – Angular UI-Grid:过滤后如何获取总项数

    提前致谢:)你应该避免使用jQuery并与API进行交互.首先需要在网格创建事件中保存对API的引用.您应该已经知道总行数.您可以使用以下命令获取可见/已过滤行数:要么您可以使用以下命令获取所选行的数量:

  10. angularjs – 迁移gulp进程以包含typescript

    或者我应该使用tsc作为我的主要构建工具,让它解决依赖关系,创建映射文件并制作捆绑包?

返回
顶部