——-> 原文地址

什么是声明文件

通常我们会把声明语句放到一个单独的文件(jQuery.d.ts)中,这就是声明文件

  1. // src/jQuery.d.ts
  2. declare var jQuery: (selector: string) => any;


声明文件必需以 .d.ts 为后缀。*
一般来说,ts 会解析项目中所有的 `
.ts文件,当然也包含以.d.ts结尾的文件。所以当我们将jQuery.d.ts放到项目中时,其他所有*.ts文件就都可以获得jQuery` 的类型定义了

假如仍然无法解析,那么可以检查下 tsconfig.json 中的 filesincludeexclude 配置,确保其包含了 jQuery.d.ts文件。
**

第三方声明文件

当然,jQuery 的声明文件不需要我们定义了,社区已经帮我们定义好了:jQuery in DefinitelyTyped
我们可以直接下载下来使用,但是更推荐的是使用 @types 统一管理第三方库的声明文件
@types 的使用方式很简单,直接用 npm 安装对应的声明模块即可,以 jQuery 举例:

  1. npm install @types/jquery --save-dev

书写声明文件

当一个第三方库没有提供声明文件时,我们就需要自己书写声明文件了。前面只介绍了最简单的声明文件内容,而真正书写一个声明文件并不是一件简单的事。以下会详细介绍如何书写声明文件。
在不同的场景下,声明文件的内容和使用方式会有所区别。
库的使用场景主要有以下几种:

  • 全局变量:通过 <script> 标签引入第三方库,注入全局变量
  • npm 包:通过 import foo from 'foo' 导入,符合 ES6 模块规范
  • UMD 库:既可以通过 <script> 标签引入,又可以通过 import 导入
  • 模块插件:通过 import 导入后,可以改变另一个模块的结构
  • 直接扩展全局变量:通过 <script> 标签引入后,改变一个全局变量的结构。比如为 String.prototype 新增了一个方法
  • 通过导入扩展全局变量:通过 import 导入后,可以改变一个全局变量的结构

**

全局变量

全局变量是最简单的一种场景,之前举的例子就是通过 <script> 标签引入 jQuery,注入全局变量 $jQuery

使用全局变量的声明文件时,如果是以 npm install @types/xxx --save-dev 安装的,则不需要任何配置。如果是将声明文件直接存放于当前项目中,则建议和其他源码一起放到 src 目录下(或者对应的源码目录下)

如果没有生效,可以检查下 tsconfig.json 中的 filesincludeexclude 配置,确保其包含了 jQuery.d.ts 文件。

全局变量的声明文件主要有以下几种语法:

  • declare var 声明全局变量
  • declare function 声明全局方法
  • declare class 声明全局类
  • declare enum 声明全局枚举类型
  • declare namespace 声明全局对象(含有子属性)
  • interfacetype 声明全局类型


一般来说,全局变量都是禁止修改的常量,所以大部分情况都应该使用 const 而不是 varlet

declare namespace

**
namespace 是 ts 早期时为了解决模块化而创造的关键字,中文称为命名空间。

由于历史遗留原因,在早期还没有 ES6 的时候,ts 提供了一种模块化方案,使用 module 关键字表示内部模块。但由于后来 ES6 也使用了 module 关键字,ts 为了兼容 ES6,使用 namespace 替代了自己的 module,更名为命名空间。

随着 ES6 的广泛应用,现在已经不建议再使用 ts 中的 namespace,而推荐使用 ES6 的模块化方案了,故我们不再需要学习 namespace 的使用了。

namespace 被淘汰了,但是在声明文件中,declare namespace 还是比较常用的,它用来表示全局变量是一个对象,包含很多子属性。

比如 jQuery 是一个全局变量,它是一个对象,提供了一个 jQuery.ajax 方法可以调用,那么我们就应该使用 declare namespace jQuery 来声明这个拥有多个子属性的全局变量。

  1. declare namespace jQuery {
  2. function ajax(url: string, settings?: any): void;
  3. }
  4. jQuery.ajax('/api/get_something');

注意,在 declare namespace 内部,我们直接使用 function ajax 来声明函数,而不是使用 declare function ajax。类似的,也可以使用 constclassenum 等语句:

  1. declare namespace jQuery {
  2. function ajax(url: string, settings?: any): void;
  3. const version: number;
  4. class Event {
  5. blur(eventType: EventType): void
  6. }
  7. enum EventType {
  8. CustomClick
  9. }
  10. }
  11. jQuery.ajax('/api/get_something');
  12. console.log(jQuery.version);
  13. const e = new jQuery.Event();
  14. e.blur(jQuery.EventType.CustomClick);

type和interface<br />

除了全局变量之外,有一些类型我们可能也希望能暴露出来。在类型声明文件中,我们可以直接使用 interfacetype 来声明一个全局的类型

  1. // src/jQuery.d.ts
  2. interface AjaxSettings {
  3. method?: 'GET' | 'POST'
  4. data?: any;
  5. }
  6. declare namespace jQuery {
  7. function ajax(url: string, settings?: AjaxSettings): void;
  8. }

这样的话,在其他文件中也可以使用这个接口了

  1. let settings: AjaxSettings = {
  2. method: 'POST',
  3. data: {
  4. name: 'foo'
  5. }
  6. };
  7. jQuery.ajax('/api/post_something', settings);

防止命名冲突

暴露在最外层的 interfacetype 会作为全局类型作用于整个项目中,我们应该尽可能的减少全局变量或全局类型的数量。故应该将他们放到 namespace 下:

  1. declare namespace jQuery {
  2. interface AjaxSettings {
  3. method?: 'GET' | 'POST'
  4. data?: any;
  5. }
  6. function ajax(url: string, settings?: AjaxSettings): void;
  7. }

声明合并

假如 jQuery 既是一个函数,可以直接被调用 jQuery('#foo'),又是一个对象,拥有子属性 jQuery.ajax()(事实确实如此),则我们可以组合多个声明语句,它们会不冲突的合并起来:

  1. declare function jQuery(selector: string): any;
  2. declare namespace jQuery {
  3. function ajax(url: string, settings?: any): void;
  4. }
  5. jQuery('#foo');
  6. jQuery.ajax('/api/get_something');

npm 包

一般我们通过 import foo from 'foo' 导入一个 npm 包,这是符合 ES6 模块规范的。

在我们尝试给一个 npm 包创建声明文件之前,首先看看它的声明文件是否已经存在。一般来说,npm 包的声明文件可能存在于两个地方:

  1. 与该 npm 包绑定在一起。判断依据是 package.json 中有 types 字段,或者有一个 index.d.ts 声明文件。这种模式不需要额外安装其他包,是最为推荐的,所以以后我们自己创建 npm 包的时候,最好也将声明文件与 npm 包绑定在一起。

**

  1. 发布到了 @types 里。只要尝试安装一下对应的包就知道是否存在,安装命令是 npm install @types/foo --save-dev。这种模式一般是由于 npm 包的维护者没有提供声明文件,所以只能由其他人将声明文件发布到 @types 里了。

**
假如以上两种方式都没有找到对应的声明文件,那么我们就需要自己为它写声明文件了。由于是通过 import 语句导入的模块,所以声明文件存放的位置也有所约束,一般有两种方案:

  1. 创建一个 node_modules/@types/foo/index.d.ts 文件,存放 foo 模块的声明文件。这种方式不需要额外的配置,但是 node_modules 目录不稳定,代码也没有被保存到仓库中,无法回溯版本,有不小心被删除的风险。

**

  1. 创建一个 types 目录,专门用来管理自己写的声明文件,将 foo 的声明文件放到 types/foo/index.d.ts 中。这种方式需要配置下 tsconfig.jsonpathsbaseUrl 字段。


目录结构:**

  1. /path/to/project
  2. ├── README.md
  3. ├── src
  4. | └── index.ts
  5. ├── types
  6. | └── foo
  7. | └── index.d.ts
  8. └── tsconfig.json

tsconfig.json 中的配置:

  1. {
  2. "compilerOptions": {
  3. "module": "commonjs",
  4. "baseUrl": "./",
  5. "paths": {
  6. "*" : ["types/*"]
  7. }
  8. }
  9. }

如此配置之后,通过 import 导入 foo 的时候,也会去 types 目录下寻找对应的模块的声明文件了。

注意 module 配置可以有很多种选项,不同的选项会影响模块的导入导出模式。这里我们使用了 commonjs 这个最常用的选项,后面的教程也都默认使用的这个选项。