当前位置: 首页 > news >正文

网站正在建设中的素材动图西安大雁塔的历史简介

网站正在建设中的素材动图,西安大雁塔的历史简介,长春火车站是哪个站,互联网前端开发declare var 声明全局变量declare function 声明全局方法declare class 声明全局类declare enum 声明全局枚举类型declare namespace 声明#xff08;含有子属性的#xff09;全局对象interface 和 type 声明全局类型export 导出变量export namespace 导出#xff08;含有子…declare var 声明全局变量declare function 声明全局方法declare class 声明全局类declare enum 声明全局枚举类型declare namespace 声明含有子属性的全局对象interface 和 type 声明全局类型export 导出变量export namespace 导出含有子属性的对象export default ES6 默认导出export  commonjs 导出模块export as namespace UMD 库声明全局变量declare global 扩展全局变量declare module 扩展模块/// reference / 三斜线指令 什么是声明语句§ 假如我们想使用第三方库 jQuery一种常见的方式是在 html 中通过 script 标签引入 jQuery然后就可以使用全局变量 $ 或 jQuery 了。 我们通常这样获取一个 id 是 foo 的元素 $(#foo); // or jQuery(#foo);但是在 ts 中编译器并不知道 $ 或 jQuery 是什么东西1 jQuery(#foo); // ERROR: Cannot find name jQuery.这时我们需要使用 declare var 来定义它的类型2 declare var jQuery: (selector: string) any;jQuery(#foo);上例中declare var 并没有真的定义一个变量只是定义了全局变量 jQuery 的类型仅仅会用于编译时的检查在编译结果中会被删除。它编译结果是 jQuery(#foo);除了 declare var 之外还有其他很多种声明语句将会在后面详细介绍。 什么是声明文件§ 通常我们会把声明语句放到一个单独的文件jQuery.d.ts中这就是声明文件3 // src/jQuery.d.tsdeclare var jQuery: (selector: string) any;// src/index.tsjQuery(#foo);声明文件必需以 .d.ts 为后缀。 一般来说ts 会解析项目中所有的 *.ts 文件当然也包含以 .d.ts 结尾的文件。所以当我们将 jQuery.d.ts 放到项目中时其他所有 *.ts 文件就都可以获得 jQuery 的类型定义了。 /path/to/project ├── src | ├── index.ts | └── jQuery.d.ts └── tsconfig.json假如仍然无法解析那么可以检查下 tsconfig.json 中的 files、include 和 exclude 配置确保其包含了 jQuery.d.ts 文件。 这里只演示了全局变量这种模式的声明文件假如是通过模块导入的方式使用第三方库的话那么引入声明文件又是另一种方式了将会在后面详细介绍。 第三方声明文件§ 当然jQuery 的声明文件不需要我们定义了社区已经帮我们定义好了jQuery in DefinitelyTyped。 我们可以直接下载下来使用但是更推荐的是使用 types 统一管理第三方库的声明文件。 types 的使用方式很简单直接用 npm 安装对应的声明模块即可以 jQuery 举例 npm install types/jquery --save-dev可以在这个页面搜索你需要的声明文件。 书写声明文件§ 当一个第三方库没有提供声明文件时我们就需要自己书写声明文件了。前面只介绍了最简单的声明文件内容而真正书写一个声明文件并不是一件简单的事以下会详细介绍如何书写声明文件。 在不同的场景下声明文件的内容和使用方式会有所区别。 库的使用场景主要有以下几种 全局变量通过 script 标签引入第三方库注入全局变量npm 包通过 import foo from foo 导入符合 ES6 模块规范UMD 库既可以通过 script 标签引入又可以通过 import 导入直接扩展全局变量通过 script 标签引入后改变一个全局变量的结构在 npm 包或 UMD 库中扩展全局变量引用 npm 包或 UMD 库后改变一个全局变量的结构模块插件通过 script 或 import 导入后改变另一个模块的结构 全局变量§ 全局变量是最简单的一种场景之前举的例子就是通过 script 标签引入 jQuery注入全局变量 $ 和 jQuery。 使用全局变量的声明文件时如果是以 npm install types/xxx --save-dev 安装的则不需要任何配置。如果是将声明文件直接存放于当前项目中则建议和其他源码一起放到 src 目录下或者对应的源码目录下 /path/to/project ├── src | ├── index.ts | └── jQuery.d.ts └── tsconfig.json如果没有生效可以检查下 tsconfig.json 中的 files、include 和 exclude 配置确保其包含了 jQuery.d.ts 文件。 全局变量的声明文件主要有以下几种语法 declare var 声明全局变量declare function 声明全局方法declare class 声明全局类declare enum 声明全局枚举类型declare namespace 声明含有子属性的全局对象interface 和 type 声明全局类型 declare var§ 在所有的声明语句中declare var 是最简单的如之前所学它能够用来定义一个全局变量的类型。与其类似的还有 declare let 和 declare const使用 let 与使用 var 没有什么区别 // src/jQuery.d.tsdeclare let jQuery: (selector: string) any;// src/index.tsjQuery(#foo); // 使用 declare let 定义的 jQuery 类型允许修改这个全局变量 jQuery function(selector) {return document.querySelector(selector); };而当我们使用 const 定义时表示此时的全局变量是一个常量不允许再去修改它的值了4 // src/jQuery.d.tsdeclare const jQuery: (selector: string) any;jQuery(#foo); // 使用 declare const 定义的 jQuery 类型禁止修改这个全局变量 jQuery function(selector) {return document.querySelector(selector); }; // ERROR: Cannot assign to jQuery because it is a constant or a read-only property.一般来说全局变量都是禁止修改的常量所以大部分情况都应该使用 const 而不是 var 或 let。 需要注意的是声明语句中只能定义类型切勿在声明语句中定义具体的实现5 declare const jQuery function(selector) {return document.querySelector(selector); }; // ERROR: An implementation cannot be declared in ambient contexts.declare function§ declare function 用来定义全局函数的类型。jQuery 其实就是一个函数所以也可以用 function 来定义 // src/jQuery.d.tsdeclare function jQuery(selector: string): any;// src/index.tsjQuery(#foo);在函数类型的声明语句中函数重载也是支持的6 // src/jQuery.d.tsdeclare function jQuery(selector: string): any; declare function jQuery(domReadyCallback: () any): any;// src/index.tsjQuery(#foo); jQuery(function() {alert(Dom Ready!); });declare class§ 当全局变量是一个类的时候我们用 declare class 来定义它的类型7 // src/Animal.d.tsdeclare class Animal {name: string;constructor(name: string);sayHi(): string; }// src/index.tslet cat new Animal(Tom);同样的declare class 语句也只能用来定义类型不能用来定义具体的实现比如定义 sayHi 方法的具体实现则会报错 // src/Animal.d.tsdeclare class Animal {name: string;constructor(name: string);sayHi() {return My name is ${this.name};};// ERROR: An implementation cannot be declared in ambient contexts. }declare enum§ 使用 declare enum 定义的枚举类型也称作外部枚举Ambient Enums举例如下8 // src/Directions.d.tsdeclare enum Directions {Up,Down,Left,Right }// src/index.tslet directions [Directions.Up, Directions.Down, Directions.Left, Directions.Right];与其他全局变量的类型声明一致declare enum 仅用来定义类型而不是具体的值。 Directions.d.ts 仅仅会用于编译时的检查声明文件里的内容在编译结果中会被删除。它编译结果是 var directions [Directions.Up, Directions.Down, Directions.Left, Directions.Right];其中 Directions 是由第三方库定义好的全局变量。 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 来声明这个拥有多个子属性的全局变量。 // src/jQuery.d.tsdeclare namespace jQuery {function ajax(url: string, settings?: any): void; }// src/index.tsjQuery.ajax(/api/get_something);注意在 declare namespace 内部我们直接使用 function ajax 来声明函数而不是使用 declare function ajax。类似的也可以使用 const, class, enum 等语句9 // src/jQuery.d.tsdeclare namespace jQuery {function ajax(url: string, settings?: any): void;const version: number;class Event {blur(eventType: EventType): void}enum EventType {CustomClick} }// src/index.tsjQuery.ajax(/api/get_something); console.log(jQuery.version); const e new jQuery.Event(); e.blur(jQuery.EventType.CustomClick);嵌套的命名空间§ 如果对象拥有深层的层级则需要用嵌套的 namespace 来声明深层的属性的类型10 // src/jQuery.d.tsdeclare namespace jQuery {function ajax(url: string, settings?: any): void;namespace fn {function extend(object: any): void;} }// src/index.tsjQuery.ajax(/api/get_something); jQuery.fn.extend({check: function() {return this.each(function() {this.checked true;});} });假如 jQuery 下仅有 fn 这一个属性没有 ajax 等其他属性或方法则可以不需要嵌套 namespace11 // src/jQuery.d.tsdeclare namespace jQuery.fn {function extend(object: any): void; }// src/index.tsjQuery.fn.extend({check: function() {return this.each(function() {this.checked true;});} });interface 和 type§ 除了全局变量之外可能有一些类型我们也希望能暴露出来。在类型声明文件中我们可以直接使用 interface 或 type 来声明一个全局的接口或类型12 // src/jQuery.d.tsinterface AjaxSettings {method?: GET | POSTdata?: any; } declare namespace jQuery {function ajax(url: string, settings?: AjaxSettings): void; }这样的话在其他文件中也可以使用这个接口或类型了 // src/index.tslet settings: AjaxSettings {method: POST,data: {name: foo} }; jQuery.ajax(/api/post_something, settings);type 与 interface 类似不再赘述。 防止命名冲突§ 暴露在最外层的 interface 或 type 会作为全局类型作用于整个项目中我们应该尽可能的减少全局变量或全局类型的数量。故最好将他们放到 namespace 下13 // src/jQuery.d.tsdeclare namespace jQuery {interface AjaxSettings {method?: GET | POSTdata?: any;}function ajax(url: string, settings?: AjaxSettings): void; }注意在使用这个 interface 的时候也应该加上 jQuery 前缀 // src/index.tslet settings: jQuery.AjaxSettings {method: POST,data: {name: foo} }; jQuery.ajax(/api/post_something, settings);声明合并§ 假如 jQuery 既是一个函数可以直接被调用 jQuery(#foo)又是一个对象拥有子属性 jQuery.ajax()事实确实如此那么我们可以组合多个声明语句它们会不冲突的合并起来14 // src/jQuery.d.tsdeclare function jQuery(selector: string): any; declare namespace jQuery {function ajax(url: string, settings?: any): void; }// src/index.tsjQuery(#foo); jQuery.ajax(/api/get_something);关于声明合并的更多用法可以查看声明合并章节。 npm 包§ 一般我们通过 import foo from foo 导入一个 npm 包这是符合 ES6 模块规范的。 在我们尝试给一个 npm 包创建声明文件之前需要先看看它的声明文件是否已经存在。一般来说npm 包的声明文件可能存在于两个地方 与该 npm 包绑定在一起。判断依据是 package.json 中有 types 字段或者有一个 index.d.ts 声明文件。这种模式不需要额外安装其他包是最为推荐的所以以后我们自己创建 npm 包的时候最好也将声明文件与 npm 包绑定在一起。发布到 types 里。我们只需要尝试安装一下对应的 types 包就知道是否存在该声明文件安装命令是 npm install types/foo --save-dev。这种模式一般是由于 npm 包的维护者没有提供声明文件所以只能由其他人将声明文件发布到 types 里了。 假如以上两种方式都没有找到对应的声明文件那么我们就需要自己为它写声明文件了。由于是通过 import 语句导入的模块所以声明文件存放的位置也有所约束一般有两种方案 创建一个 node_modules/types/foo/index.d.ts 文件存放 foo 模块的声明文件。这种方式不需要额外的配置但是 node_modules 目录不稳定代码也没有被保存到仓库中无法回溯版本有不小心被删除的风险故不太建议用这种方案一般只用作临时测试。创建一个 types 目录专门用来管理自己写的声明文件将 foo 的声明文件放到 types/foo/index.d.ts 中。这种方式需要配置下 tsconfig.json 中的 paths 和 baseUrl 字段。 目录结构 /path/to/project ├── src | └── index.ts ├── types | └── foo | └── index.d.ts └── tsconfig.jsontsconfig.json 内容 {compilerOptions: {module: commonjs,baseUrl: ./,paths: {*: [types/*]}} }如此配置之后通过 import 导入 foo 的时候也会去 types 目录下寻找对应的模块的声明文件了。 注意 module 配置可以有很多种选项不同的选项会影响模块的导入导出模式。这里我们使用了 commonjs 这个最常用的选项后面的教程也都默认使用的这个选项。 不管采用了以上两种方式中的哪一种我都强烈建议大家将书写好的声明文件通过给第三方库发 pull request或者直接提交到 types 里发布到开源社区中享受了这么多社区的优秀的资源就应该在力所能及的时候给出一些回馈。只有所有人都参与进来才能让 ts 社区更加繁荣。 npm 包的声明文件主要有以下几种语法 export 导出变量export namespace 导出含有子属性的对象export default ES6 默认导出export  commonjs 导出模块 export§ npm 包的声明文件与全局变量的声明文件有很大区别。在 npm 包的声明文件中使用 declare 不再会声明一个全局变量而只会在当前文件中声明一个局部变量。只有在声明文件中使用 export 导出然后在使用方 import 导入后才会应用到这些类型声明。 export 的语法与普通的 ts 中的语法类似区别仅在于声明文件中禁止定义具体的实现15 // types/foo/index.d.tsexport const name: string; export function getName(): string; export class Animal {constructor(name: string);sayHi(): string; } export enum Directions {Up,Down,Left,Right } export interface Options {data: any; }对应的导入和使用模块应该是这样 // src/index.tsimport { name, getName, Animal, Directions, Options } from foo;console.log(name); let myName getName(); let cat new Animal(Tom); let directions [Directions.Up, Directions.Down, Directions.Left, Directions.Right]; let options: Options {data: {name: foo} };混用 declare 和 export§ 我们也可以使用 declare 先声明多个变量最后再用 export 一次性导出。上例的声明文件可以等价的改写为16 // types/foo/index.d.tsdeclare const name: string; declare function getName(): string; declare class Animal {constructor(name: string);sayHi(): string; } declare enum Directions {Up,Down,Left,Right } interface Options {data: any; }export { name, getName, Animal, Directions, Options };注意与全局变量的声明文件类似interface 前是不需要 declare 的。 export namespace§ 与 declare namespace 类似export namespace 用来导出一个拥有子属性的对象17 // types/foo/index.d.tsexport namespace foo {const name: string;namespace bar {function baz(): string;} }// src/index.tsimport { foo } from foo;console.log(foo.name); foo.bar.baz();export default§ 在 ES6 模块系统中使用 export default 可以导出一个默认值使用方可以用 import foo from foo 而不是 import { foo } from foo 来导入这个默认值。 在类型声明文件中export default 用来导出默认值的类型18 // types/foo/index.d.tsexport default function foo(): string;// src/index.tsimport foo from foo;foo();注意只有 function、class 和 interface 可以直接默认导出其他的变量需要先定义出来再默认导出19 // types/foo/index.d.tsexport default enum Directions { // ERROR: Expression expected.Up,Down,Left,Right }上例中 export default enum 是错误的语法需要使用 declare enum 定义出来然后使用 export default 导出 // types/foo/index.d.tsdeclare enum Directions {Up,Down,Left,Right }export default Directions;针对这种默认导出我们一般会将导出语句放在整个声明文件的最前面20 // types/foo/index.d.tsexport default Directions;declare enum Directions {Up,Down,Left,Right }export § 在 commonjs 规范中我们用以下方式来导出一个模块 // 整体导出 module.exports foo; // 单个导出 exports.bar bar;在 ts 中针对这种模块导出有多种方式可以导入第一种方式是 const ... require // 整体导入 const foo require(foo); // 单个导入 const bar require(foo).bar;第二种方式是 import ... from注意针对整体导出需要使用 import * as 来导入 // 整体导入 import * as foo from foo; // 单个导入 import { bar } from foo;第三种方式是 import ... require这也是 ts 官方推荐的方式 // 整体导入 import foo require(foo); // 单个导入 import bar foo.bar;对于这种使用 commonjs 规范的库假如要为它写类型声明文件的话就需要使用到 export  这种语法了21 // types/foo/index.d.tsexport foo;declare function foo(): string; declare namespace foo {const bar: number; }需要注意的是上例中使用了 export  之后就不能再单个导出 export { bar } 了。所以我们通过声明合并使用 declare namespace foo 来将 bar 合并到 foo 里。 准确地讲export  不仅可以用在声明文件中也可以用在普通的 ts 文件中。实际上import ... require 和 export  都是 ts 为了兼容 AMD 规范和 commonjs 规范而创立的新语法由于并不常用也不推荐使用所以这里就不详细介绍了感兴趣的可以看官方文档。 由于很多第三方库是 commonjs 规范的所以声明文件也就不得不用到 export  这种语法了。但是还是需要再强调下相比与 export 我们更推荐使用 ES6 标准的 export default 和 export。 UMD 库§ 既可以通过 script 标签引入又可以通过 import 导入的库称为 UMD 库。相比于 npm 包的类型声明文件我们需要额外声明一个全局变量为了实现这种方式ts 提供了一个新语法 export as namespace。 export as namespace§ 一般使用 export as namespace 时都是先有了 npm 包的声明文件再基于它添加一条 export as namespace 语句即可将声明好的一个变量声明为全局变量举例如下22 // types/foo/index.d.tsexport as namespace foo; export foo;declare function foo(): string; declare namespace foo {const bar: number; }当然它也可以与 export default 一起使用 // types/foo/index.d.tsexport as namespace foo; export default foo;declare function foo(): string; declare namespace foo {const bar: number; }直接扩展全局变量§ 有的第三方库扩展了一个全局变量可是此全局变量的类型却没有相应的更新过来就会导致 ts 编译错误此时就需要扩展全局变量的类型。比如扩展 String 类型23 interface String {prependHello(): string; }foo.prependHello();通过声明合并使用 interface String 即可给 String 添加属性或方法。 也可以使用 declare namespace 给已有的命名空间添加类型声明24 // types/jquery-plugin/index.d.tsdeclare namespace JQuery {interface CustomOptions {bar: string;} }interface JQueryStatic {foo(options: JQuery.CustomOptions): string; }// src/index.tsjQuery.foo({bar: });在 npm 包或 UMD 库中扩展全局变量§ 如之前所说对于一个 npm 包或者 UMD 库的声明文件只有 export 导出的类型声明才能被导入。所以对于 npm 包或 UMD 库如果导入此库之后会扩展全局变量则需要使用另一种语法在声明文件中扩展全局变量的类型那就是 declare global。 declare global§ 使用 declare global 可以在 npm 包或者 UMD 库的声明文件中扩展全局变量的类型25 // types/foo/index.d.tsdeclare global {interface String {prependHello(): string;} }export {};// src/index.tsbar.prependHello();注意即使此声明文件不需要导出任何东西仍然需要导出一个空对象用来告诉编译器这是一个模块的声明文件而不是一个全局变量的声明文件。 模块插件§ 有时通过 import 导入一个模块插件可以改变另一个原有模块的结构。此时如果原有模块已经有了类型声明文件而插件模块没有类型声明文件就会导致类型不完整缺少插件部分的类型。ts 提供了一个语法 declare module它可以用来扩展原有模块的类型。 declare module§ 如果是需要扩展原有模块的话需要在类型声明文件中先引用原有模块再使用 declare module 扩展原有模块26 // types/moment-plugin/index.d.tsimport * as moment from moment;declare module moment {export function foo(): moment.CalendarKey; }// src/index.tsimport * as moment from moment; import moment-plugin;moment.foo();declare module 也可用于在一个文件中一次性声明多个模块的类型27 // types/foo-bar.d.tsdeclare module foo {export interface Foo {foo: string;} }declare module bar {export function bar(): string; }// src/index.tsimport { Foo } from foo; import * as bar from bar;let f: Foo; bar.bar();声明文件中的依赖§ 一个声明文件有时会依赖另一个声明文件中的类型比如在前面的 declare module 的例子中我们就在声明文件中导入了 moment并且使用了 moment.CalendarKey 这个类型 // types/moment-plugin/index.d.tsimport * as moment from moment;declare module moment {export function foo(): moment.CalendarKey; }除了可以在声明文件中通过 import 导入另一个声明文件中的类型之外还有一个语法也可以用来导入另一个声明文件那就是三斜线指令。 三斜线指令§ 与 namespace 类似三斜线指令也是 ts 在早期版本中为了描述模块之间的依赖关系而创造的语法。随着 ES6 的广泛应用现在已经不建议再使用 ts 中的三斜线指令来声明模块之间的依赖关系了。 但是在声明文件中它还是有一定的用武之地。 类似于声明文件中的 import它可以用来导入另一个声明文件。与 import 的区别是当且仅当在以下几个场景下我们才需要使用三斜线指令替代 import 当我们在书写一个全局变量的声明文件时当我们需要依赖一个全局变量的声明文件时 书写一个全局变量的声明文件§ 这些场景听上去很拗口但实际上很好理解——在全局变量的声明文件中是不允许出现 import, export 关键字的。一旦出现了那么他就会被视为一个 npm 包或 UMD 库就不再是全局变量的声明文件了。故当我们在书写一个全局变量的声明文件时如果需要引用另一个库的类型那么就必须用三斜线指令了28 // types/jquery-plugin/index.d.ts/// reference typesjquery /declare function foo(options: JQuery.AjaxSettings): string;// src/index.tsfoo({});三斜线指令的语法如上/// 后面使用 xml 的格式添加了对 jquery 类型的依赖这样就可以在声明文件中使用 JQuery.AjaxSettings 类型了。 注意三斜线指令必须放在文件的最顶端三斜线指令的前面只允许出现单行或多行注释。 依赖一个全局变量的声明文件§ 在另一个场景下当我们需要依赖一个全局变量的声明文件时由于全局变量不支持通过 import 导入当然也就必须使用三斜线指令来引入了29 // types/node-plugin/index.d.ts/// reference typesnode /export function foo(p: NodeJS.Process): string;// src/index.tsimport { foo } from node-plugin;foo(global.process);在上面的例子中我们通过三斜线指引入了 node 的类型然后在声明文件中使用了 NodeJS.Process 这个类型。最后在使用到 foo 的时候传入了 node 中的全局变量 process。 由于引入的 node 中的类型都是全局变量的类型它们是没有办法通过 import 来导入的所以这种场景下也只能通过三斜线指令来引入了。 以上两种使用场景下都是由于需要书写或需要依赖全局变量的声明文件所以必须使用三斜线指令。在其他的一些不是必要使用三斜线指令的情况下就都需要使用 import 来导入。 拆分声明文件§ 当我们的全局变量的声明文件太大时可以通过拆分为多个文件然后在一个入口文件中将它们一一引入来提高代码的可维护性。比如 jQuery 的声明文件就是这样的 // node_modules/types/jquery/index.d.ts/// reference typessizzle / /// reference pathJQueryStatic.d.ts / /// reference pathJQuery.d.ts / /// reference pathmisc.d.ts / /// reference pathlegacy.d.ts /export jQuery;其中用到了 types 和 path 两种不同的指令。它们的区别是types 用于声明对另一个库的依赖而 path 用于声明对另一个文件的依赖。 上例中sizzle 是与 jquery 平行的另一个库所以需要使用 typessizzle 来声明对它的依赖。而其他的三斜线指令就是将 jquery 的声明拆分到不同的文件中了然后在这个入口文件中使用 pathfoo 将它们一一引入。 其他三斜线指令§ 除了这两种三斜线指令之外还有其他的三斜线指令比如 /// reference no-default-libtrue/, /// amd-module / 等但它们都是废弃的语法故这里就不介绍了详情可见官网。 自动生成声明文件§ 如果库的源码本身就是由 ts 写的那么在使用 tsc 脚本将 ts 编译为 js 的时候添加 declaration 选项就可以同时也生成 .d.ts 声明文件了。 我们可以在命令行中添加 --declaration简写 -d或者在 tsconfig.json 中添加 declaration 选项。这里以 tsconfig.json 为例 {compilerOptions: {module: commonjs,outDir: lib,declaration: true,} }上例中我们添加了 outDir 选项将 ts 文件的编译结果输出到 lib 目录下然后添加了 declaration 选项设置为 true表示将会由 ts 文件自动生成 .d.ts 声明文件也会输出到 lib 目录下。 运行 tsc 之后目录结构如下30 /path/to/project ├── lib | ├── bar | | ├── index.d.ts | | └── index.js | ├── index.d.ts | └── index.js ├── src | ├── bar | | └── index.ts | └── index.ts ├── package.json └── tsconfig.json在这个例子中src 目录下有两个 ts 文件分别是 src/index.ts 和 src/bar/index.ts它们被编译到 lib 目录下的同时也会生成对应的两个声明文件 lib/index.d.ts 和 lib/bar/index.d.ts。它们的内容分别是 // src/index.tsexport * from ./bar;export default function foo() {return foo; }// src/bar/index.tsexport function bar() {return bar; }// lib/index.d.tsexport * from ./bar; export default function foo(): string;// lib/bar/index.d.tsexport declare function bar(): string;可见自动生成的声明文件基本保持了源码的结构而将具体实现去掉了生成了对应的类型声明。 使用 tsc 自动生成声明文件时每个 ts 文件都会对应一个 .d.ts 声明文件。这样的好处是使用方不仅可以在使用 import foo from foo 导入默认的模块时获得类型提示还可以在使用 import bar from foo/lib/bar 导入一个子模块时也获得对应的类型提示。 除了 declaration 选项之外还有几个选项也与自动生成声明文件有关这里只简单列举出来不做详细演示了 declarationDir 设置生成 .d.ts 文件的目录declarationMap 对每个 .d.ts 文件都生成对应的 .d.ts.mapsourcemap文件emitDeclarationOnly 仅生成 .d.ts 文件不生成 .js 文件 发布声明文件§ 当我们为一个库写好了声明文件之后下一步就是将它发布出去了。 此时有两种方案 将声明文件和源码放在一起将声明文件发布到 types 下 这两种方案中优先选择第一种方案。保持声明文件与源码在一起使用时就不需要额外增加单独的声明文件库的依赖了而且也能保证声明文件的版本与源码的版本保持一致。 仅当我们在给别人的仓库添加类型声明文件但原作者不愿意合并 pull request 时才需要使用第二种方案将声明文件发布到 types 下。 将声明文件和源码放在一起§ 如果声明文件是通过 tsc 自动生成的那么无需做任何其他配置只需要把编译好的文件也发布到 npm 上使用方就可以获取到类型提示了。 如果是手动写的声明文件那么需要满足以下条件之一才能被正确的识别 给 package.json 中的 types 或 typings 字段指定一个类型声明文件地址在项目根目录下编写一个 index.d.ts 文件针对入口文件package.json 中的 main 字段指定的入口文件编写一个同名不同后缀的 .d.ts 文件 第一种方式是给 package.json 中的 types 或 typings 字段指定一个类型声明文件地址。比如 {name: foo,version: 1.0.0,main: lib/index.js,types: foo.d.ts, }指定了 types 为 foo.d.ts 之后导入此库的时候就会去找 foo.d.ts 作为此库的类型声明文件了。 typings 与 types 一样只是另一种写法。 如果没有指定 types 或 typings那么就会在根目录下寻找 index.d.ts 文件将它视为此库的类型声明文件。 如果没有找到 index.d.ts 文件那么就会寻找入口文件package.json 中的 main 字段指定的入口文件是否存在对应同名不同后缀的 .d.ts 文件。 比如 package.json 是这样时 {name: foo,version: 1.0.0,main: lib/index.js }就会先识别 package.json 中是否存在 types 或 typings 字段。发现不存在那么就会寻找是否存在 index.d.ts 文件。如果还是不存在那么就会寻找是否存在 lib/index.d.ts 文件。假如说连 lib/index.d.ts 都不存在的话就会被认为是一个没有提供类型声明文件的库了。 有的库为了支持导入子模块比如 import bar from foo/lib/bar就需要额外再编写一个类型声明文件 lib/bar.d.ts 或者 lib/bar/index.d.ts这与自动生成声明文件类似一个库中同时包含了多个类型声明文件。 将声明文件发布到 types 下§ 如果我们是在给别人的仓库添加类型声明文件但原作者不愿意合并 pull request那么就需要将声明文件发布到 types 下。 与普通的 npm 模块不同types 是统一由 DefinitelyTyped 管理的。要将声明文件发布到 types 下就需要给 DefinitelyTyped 创建一个 pull-request其中包含了类型声明文件测试代码以及 tsconfig.json 等。 pull-request 需要符合它们的规范并且通过测试才能被合并稍后就会被自动发布到 types 下。 在 DefinitelyTyped 中创建一个新的类型声明需要用到一些工具DefinitelyTyped 的文档中已经有了详细的介绍这里就不赘述了以官方文档为准。 参考文档 简介 · TypeScript 入门教程
http://www.w-s-a.com/news/294599/

相关文章:

  • 建公司网站wordpress the content
  • 网站的站点的管理系统建设银行网站注册企业
  • 长春火车站是哪个站做微商哪个网站有客源
  • 亚马逊培训费用一般多少seo专业培训课程
  • 做推文封面图网站南宁高端网站建设
  • 天津网站搜索排名做电影免费ppt模板下载网站
  • 襄樊最好网站建设价格网站建设与设计 毕业设计
  • 网站推广广告词大全集网站建设相对路径
  • 甘肃省铁路投资建设集团有限公司网站域名怎么实名认证
  • 企业网站建设的层次返利网站建设
  • 竞价单页网站制作中小企业网站建设问题
  • 响应式网站源码学校网站制作多少钱
  • 营销型网站建设需要懂什么网站建站四件套是什么
  • 廊坊哪里有制作手机网站的区块链开发语言
  • 找建设项目的网站装企工长网站开发
  • 戴尔网站建设成功方正网站制作
  • 怎么买网站域名wordpress 视频站模版
  • 厦门官网建设公司杨和关键词优化
  • 怎么做网约车seo自动优化软件下载
  • 遵义市住房和城乡建设局官方网站网站备案 自己的服务器
  • 分销系统价格多少北京网站优化平台
  • 怎样做旅游公司的网站泉州网站建设方案优化
  • 手机网站页面范例个人网站做淘宝客违规
  • 做一套网站开发多少钱SEO做得最好的网站
  • 咸宁做网站的公司那家便宜福建建设注册管理中心网站
  • 网站建设工作汇报黑科技广告推广神器
  • 淘宝做首页热点的什么网站徐州建设安全监督网站
  • 正规的镇江网站建设广州有什么好玩的东西
  • 丹阳网站设计公司网站开发 0755
  • 百度网页版浏览器网址找文网优化的技术团队