从编译后的代码,分析 Angular @Injectable 的工作原理

简介: 从编译后的代码,分析 Angular @Injectable 的工作原理

在开始本文的讨论之前,让我们先做一道多选题:


下面对于 @Injectable 的描述,哪些是正确的?


A @Injectable 装饰器应该添加到每个服务中。

B @Injectable 装饰器应该添加到任何使用依赖注入 (DI) 的服务中。

C 如果您不使用“providedIn”选项,则不需要添加 @Injectable 装饰器。

D @Injectable 装饰器与“providedIn”选项一起,意味着不应将服务添加到模块的提供者数组中。


答案是 B 和 D.


Angular 带有自己的 DI 框架,并用于 Angular 应用程序的设计以增强其模块化和效率。 @Injectable() 是任何 Angular 服务定义的重要组成部分。 有些人可能选择错误答案的原因是他们错误地判断了@Injectable 装饰器和Angular 中的DI 之间的关系,不了解这个装饰器的真正作用。

Adding @Injectable Is Not Registering the Service

对于用于 DI 的控制反转 (IOC) 容器,有两个成对的知识点需要掌握。 首先是令牌。 要向 IOC 容器注册某些服务,需要一个令牌。 令牌是注册任何服务的唯一标识符。 第二个知识点是 provider.


提供者帮助 DI 容器创建特定依赖项的实例。


在 Angular 中,使用令牌注册服务并将其传递给提供者可以通过两种不同的方式完成。


首先,可以使用特定的 @NgModule 注册服务。 该过程是通过将服务传递给提供者数组 providers 来进行注册。在下面这个例子里,使用的令牌是 typescript 类型 MyService。 这里的提供者是 useClass。 这个提供者策略,通知 Angular 框架,可以通过 new 关键字来启动某个依赖项的实例。

@NgModule({
  ...
  providers: [
    // long hand syntax
    {provide: MyService, useClass: MyService},
    // short hand syntax
    MyService
  ],
})

第二种方式是使用 @Injectable

@Injectable({
  providedIn: 'root'
})
export class MyService {
  constructor() { }
}

这里要注意区分的一点是,@Injectable 装饰器与向容器注册服务无关。 那么@Injectable 装饰器的目的是什么?

事实上,MyService 可以有一些依赖项是标准的。 例如,您可以使用 HttpClient。

import { HttpClient } from '@angular/common/http';
@Injectable()
export class MyService {
  constructor(private httpClient: HttpClient) { }
}

让我们尝试了解当您需要获取 MyService 实例时会发生什么。 当 Angular 需要创建服务时,它应该传递一个 HttpClient 的实例。 那么它怎么能做到呢?


首先,Angular 应该知道提到了哪个依赖项。 还记得您使用令牌来注册依赖项吗? 出于同样的原因,您可以向 Angular 请求该依赖项的一个实例。 因此,Angular 将调查构造函数并查看是否为令牌 HttpClient 请求了服务。 如果使用该令牌注册了服务,Angular 可以使用该令牌启动 MyService 并提供它创建的实例。


下面是编译后的 JavaScript 代码:

var MyService = /** @class */ (function () {
    function MyService(httpClient) {
        this.httpClient = httpClient;
    }
    MyService = __decorate([
        Object(_angular_core__WEBPACK_IMPORTED_MODULE_0__["Injectable"])(),
        __metadata(
          "design:paramtypes", 
          [_angular_common_http__WEBPACK_IMPORTED_MODULE_1__["HttpClient"]]
        )
    ], MyService);
    return MyService;
}());
相关文章
|
15天前
|
前端开发 大数据 数据库
🔥大数据洪流下的决战:JSF 表格组件如何做到毫秒级响应?揭秘背后的性能魔法!💪
【8月更文挑战第31天】在 Web 应用中,表格组件常用于展示和操作数据,但在大数据量下性能会成瓶颈。本文介绍在 JavaServer Faces(JSF)中优化表格组件的方法,包括数据处理、分页及懒加载等技术。通过后端分页或懒加载按需加载数据,减少不必要的数据加载和优化数据库查询,并利用缓存机制减少数据库访问次数,从而提高表格组件的响应速度和整体性能。掌握这些最佳实践对开发高性能 JSF 应用至关重要。
30 0
|
15天前
|
Java Spring
🔥JSF 与 Spring 强强联手:打造高效、灵活的 Web 应用新标杆!💪 你还不知道吗?
【8月更文挑战第31天】JavaServer Faces(JSF)与 Spring 框架是常用的 Java Web 技术。本文介绍如何整合两者,发挥各自优势,构建高效灵活的 Web 应用。首先通过 `web.xml` 和 `ContextLoaderListener` 配置 Spring 上下文,在 `applicationContext.xml` 定义 Bean。接着使用 `@Autowired` 将 Spring 管理的 Bean 注入到 JSF 管理的 Bean 中。
29 0
|
15天前
|
开发者 Java 开发框架
JSF与EJB,打造企业级应用的神器!让你的Web应用更加稳定、高效!
【8月更文挑战第31天】在现代企业级应用开发中,JSF(JavaServer Faces)与EJB(Enterprise JavaBeans)是两大核心技术。JSF作为一款基于Java的Web应用框架,以其丰富的UI组件和表单处理功能著称;EJB则专注于提供分布式事务处理及远程调用等企业级服务。两者的结合为企业应用带来了高效便捷的开发模式。下文将通过一个简单的示例展示如何利用JSF进行用户信息的输入与保存,并借助EJB实现相关业务逻辑。尽管这一组合具有明显优势,但在实际应用中还需考虑其局限性并作出合理选择。
30 0
|
15天前
|
开发者 缓存 数据库
【性能奇迹】Wicket应用的极速重生:揭秘那些让开发者心跳加速的调优秘技!
【8月更文挑战第31天】在软件开发中,性能优化是确保应用快速响应和高效运行的关键。本书《性能调优:Apache Wicket应用的速度提升秘籍》详细介绍了如何优化Apache Wicket应用,包括代码优化、资源管理、数据库查询优化、缓存策略及服务器配置等方面。通过减少不必要的组件渲染、优化SQL查询、使用缓存和调整服务器设置等方法,本书帮助开发者显著提升Wicket应用的性能,确保其在高并发和数据密集型场景下的稳定性和响应速度。
27 0
|
4月前
|
JavaScript 前端开发 编译器
Angular 中的结构指令运行时的工作原理
Angular 中的结构指令运行时的工作原理
|
4月前
|
前端开发 JavaScript
Angular 组件模版代码里使用 ngIf 进行条件渲染的例子
Angular 组件模版代码里使用 ngIf 进行条件渲染的例子
|
4月前
|
JavaScript 前端开发 API
vue的双向绑定的原理,和angular的对比
vue的双向绑定的原理,和angular的对比
88 0
|
10月前
|
JSON 搜索推荐 数据格式
Angular SSR 应用中 serverApp-state script 的工作原理介绍
Angular SSR 应用中 serverApp-state script 的工作原理介绍
|
11月前
Angular 应用里 NullInjectorError - No provider for XX 错误的一个场景和分析过程
Angular 应用里 NullInjectorError - No provider for XX 错误的一个场景和分析过程
|
12月前
|
存储 JavaScript 容器
关于 Angular 注解 @Injectable() 使用的一些误区
关于 Angular 注解 @Injectable() 使用的一些误区