理解 Angular 服务

本文写于 2021 年 3 月 29 日

什么是服务

angular 的最佳实践推荐业务逻辑要尽量分门别类的抽离为一个个的 Service。

那么到底什么是 Service 呢?

service 意为服务,是一个广义的概念。

例如:饭馆提供的是吃饭服务、澡堂提供的是洗澡服务、学校提供的是授课服务……

那么对于写代码来说,服务是什么呢?

服务一般会包括值/状态、函数或各种特性,在 Angular 中,我们使用一个 class 来包裹这些。

但实际上,服务可以不是一个类,例如 React 的 context,通过闭包得到的返回值也可以是一个服务。

归根结底,可以这么总结:把一系列高度聚合的代码放到一起,就是一个服务

对于前端来说,操心的事情其实有两件:

  1. 用户体验(动画、跳转、样式……)
  2. 业务逻辑与状态管理

Angular 推荐我们将第二点放到服务中来完成,从而令组件可以专心的负责用户体验。

比如从服务器获取数据、验证用户输入、日志……都应该放在服务,而不是特定的组件中。

当我们将这些委托给了各种可注入的服务之后,就可以将其注入到各个组件或者其他服务中进行调用。让你的代码更具适应性。

在 Angular 中,服务通常是一个单例。

服务写法

代码来自 Angular 官方文档,有删改

// 省略 @Injectable 代码
export class Logger {
log(msg: any) {
console.log(msg);
} error(msg: any) {
console.error(msg);
} warn(msg: any) {
console.warn(msg);
}
}

在 ng 组件或是 service 的构造函数中注入服务(框架会帮我们自动完成)后即可在其他方法中调用。

// 省略 @Injectable 代码
export class HeroService {
constructor(private logger: Logger) {} getHeroes() {
this.getXXXXData().then((res: any) => {
this.logger.log(`Fetched data: ${res}`);
});
}
}

原理简述

当 Angular 创建组件类的新实例时,框架会查看该组件类的 constructor,得到该组件依赖哪些服务或其它依赖项。

如果发现了某个组件依赖某个服务,框架会首先检查注入器中是否已经有了那个服务的实例。如果存在,直接返回;如果不存在,注入器就会使用以前注册的服务提供者来创造一个,并把它加入注入器中,最后把该服务返回。

提供服务

服务需要「提供」,才能使用。

1. 在服务中注册

@Injectable({
providedIn: "root",
})
export class SomeService {}

默认情况下,服务会提供给 root,也就是根。这样该服务就可以在任何 module 的组件进行注入。

这种在 @Injectable 元数据中注册提供者的方式还让 Angular 能够通过移除那些从未被用过的服务来优化大小。

2. 在 module 中注册

@NgModule({
providers: [
AService,
BService
],
...
})

你也可以提供给单独的 module,让 service 只能在该 module 使用。

3. 在组件中注册

@Component({
selector: 'app-hero-list',
templateUrl: './hero-list.component.html',
providers: [ SomeService ]
})

当你在组件级注册提供者时,你会为该组件的每一个新实例提供该服务的一个新实例!!!

所以不推荐这么做。

该在哪里注入服务

Angular 使得我们可以在任何的组件中注入服务,这会使得我们“滥用”注入。

这里我们借用一些 react-redux 中的概念——展示组件和容器组件。

在展示组件中,应该只负责定义自己需要的状态、方法,具体的依赖于外层传入。

在容器组件中,才可以注入服务进行操作。

这样能有效的降低组件对于特定服务的依赖。

(完)

理解 Angular 服务的更多相关文章

  1. (七)理解angular中的module和injector,即依赖注入

    (七)理解angular中的module和injector,即依赖注入 时间:2014-10-10 01:16:54      阅读:63060      评论:1      收藏:0      [点 ...

  2. 理解angular中的module和injector,即依赖注入

    理解angular中的module和injector,即依赖注入 依赖注入(DI)的好处不再赘言,使用过spring框架的都知道.angularjs作为前台js框架,也提供了对DI的支持,这是java ...

  3. angular服务二

    angular服务 $http 实现客户端与服务器端异步请求 get方式 test.html <!DOCTYPE html> <html lang="en"> ...

  4. angular服务一

    angular服务 [$apply()] jquery内数据绑定 要用$apply(),不然不能在js内通过angular绑定数据 <!DOCTYPE html> <html lan ...

  5. angular 服务 service factory provider constant value

    angular服务 服务是对公共代码的抽象,由于依赖注入的要求,服务都是单例的,这样我们才能到处注入它们,而不用去管理它们的生命周期. angular的服务有以下几种类型: 常量(Constant): ...

  6. 第217天:深入理解Angular双向数据绑定的原理

    一.理解angular双向数据绑定 双向绑定是新的前端框架中频繁出现的一个新词汇,也是mvvm的核心原理.angularjs五条核心信念中的数据驱动,便是由双向绑定进行完成. 那么什么是双向绑定,下面 ...

  7. Java生鲜电商平台-深入理解微服务SpringCloud各个组件的关联与架构

    Java生鲜电商平台-深入理解微服务SpringCloud各个组件的关联与架构 概述 毫无疑问,Spring Cloud是目前微服务架构领域的翘楚,无数的书籍博客都在讲解这个技术.不过大多数讲解还停留 ...

  8. 理解Angular中的$apply()以及$digest()

    $apply()和$digest()在AngularJS中是两个核心概念,但是有时候它们又让人困惑.而为了了解AngularJS的工作方式,首先需要了解$apply()和$digest()是如何工作的 ...

  9. 深入理解Angular中的$Apply()以及$Digest()

    $apply()和$digest()在AngularJS中是两个核心概念,但是有时候它们又让人困惑.而为了了解AngularJS的工作方式,首先需要了解$apply()和$digest()是如何工作的 ...

随机推荐

  1. Centos6 编译安装Python3.6

    1. 安装依赖 yum install gcc openssl-devel bzip2-devel 2. 下载Python3.6 cd /usr/src wget https://www.python ...

  2. linux文本编辑器vim详解

    vim 1.打开文件 vim [option] - file... 打开文件 +# 打开文件后,让光标处于第#行的行首 +/字符串 打开文件后,光标处于第一个被匹配到字符串的行首 -b file 二进 ...

  3. js技术之转换为大写toUpperCase()

    案例:把所有单词以空格为分割并将首字母转为大写 <!DOCTYPE html><html lang="en"><head> <meta c ...

  4. CSS入门指南-4:页面布局

    这是<CSS设计指南>的读书笔记,用于加深学习效果. display 属性 display是 CSS 中最重要的用于控制布局的属性.每个元素都有一个默认的 display 值.对于大多数元 ...

  5. 【译】HTML表单高级样式

    系列文章说明 原文 在本文中,我们将了解如何在HTML表单上使用CSS,为那些难于自定义的表单组件加以样式.如前文所述,文本框和按钮很适合使用CSS,而现在我们得来探索HTML表单样式的那些坑了. 在 ...

  6. HTML5 & CSS3 内容收集(1)

    1. HTML发展历史介绍 2. 浏览器支持 2.1 新增标签支持 在html5 中新增了很多的标签,其中包括8个新增语义结构标签.header, section, footer, aside, na ...

  7. H5的audio在ios系统的微信上不能自动播放的问题

    前几天有个需求,要在H5页面中添加背景音乐,本以为很easy,却也踩了一些坑,废话不多说,进入正题: 撸完代码测试的时候才发现在安卓手机上背景音乐可以正常播放,但在iphone里的微信和safari中 ...

  8. ES6-11学习笔记--深拷贝与浅拷贝

    Object.assign,只是进行了浅拷贝,并没有进行深拷贝. 而且会在复杂结构当中会丢失属性,如下代码: let target = { a: { b: { c: 3 }, e: 4, f: 5, ...

  9. python-筛法求素数

    [题目描述]用户输入整数n和m(1<n<m<1000),应用筛法求[n,m]范围内的所有素数. [练习要求]请给出源代码程序和运行测试结果,源代码程序要求添加必要的注释. [输入格式 ...

  10. Mysql_事务_存储过程_触发器

    一.什么是事务? 事务(Transaction),一般是指要做的或所做的事情.在计算机术语中是指访问并可能更新数据库中各种数据项的一个程序执行单元(unit).事务通常由高级数据库操纵语言或编程语言( ...