资讯专栏INFORMATION COLUMN

TypeScript 初识 - 接口

gnehc / 1628人阅读

摘要:接口的作用是声明变量的结构和方法,但不做具体的实现。这两个使用场景不同。额外的属性检查从字面上的意思看,其实就是对接口未定义的属性进行检查。上面的例子,定义了接口,它具有索引签名。它不会帮你检查类是否具有某些私有成员。

接口的作用是声明变量的结构和方法,但不做具体的实现。通常,接口会强制对所有成员进行类型检查,包括数量和类型:

</>复制代码

  1. interface Name {
  2. first: string;
  3. second: string;
  4. }
  5. let name: Name;
  6. name = {
  7. first: "John",
  8. second: "Doe"
  9. };
  10. name = {
  11. // Error: "second is missing"
  12. first: "John"
  13. };
  14. name = {
  15. // Error: "second is the wrong type"
  16. first: "John",
  17. second: 1337
  18. };
可选属性

带有可选属性的接口与普通的接口定义差不多,只是在可选属性名字定义的后面加一个 ? 符号:

</>复制代码

  1. interface Name {
  2. first?: string;
  3. second?: string;
  4. }
  5. let name: Name;
  6. name = {
  7. // 只有 first 也不会报错
  8. first: "John"
  9. };
只读属性

一些对象属性只能在对象刚刚创建的时候修改其值。可以在属性名前用 readonly 来指定只读属性:

</>复制代码

  1. interface Point {
  2. readonly x: number;
  3. readonly y: number;
  4. }
  5. let p1: Point = { x: 10, y: 20 };
  6. p1.x = 5; // error!

TypeScript 具有 ReadonlyArray 类型,它与 Array 相似,只是把所有可变方法去掉了,因此可以确保数组创建后再也不能被修改:

</>复制代码

  1. let a: number[] = [1, 2, 3, 4];
  2. let ro: ReadonlyArray = a;
  3. ro[0] = 12; // error!
  4. ro.push(5); // error!
  5. ro.length = 100; // error!
  6. a = ro; // error!

readonly 针对一个对象的属性,const 针对的是一个变量。这两个使用场景不同。

额外的属性检查

从字面上的意思看,其实就是对接口未定义的属性进行检查。

当一个变量实现接口时,变量中存在了接口未定义的属性时,TypeScript 会进行报错:

</>复制代码

  1. interface Point {
  2. x: number;
  3. y?: number;
  4. }
  5. // error!
  6. const myPoint: Point = { x: 1, z: 3 };

要想绕过检查,最简单的方法就是使用类型断言:

</>复制代码

  1. const myPoint: Point = { x: 1, z: 3 } as Point;

最佳的方法是定义接口的时候添加一个字符串索引签名:

</>复制代码

  1. interface Point {
  2. x: number;
  3. y?: number;
  4. [propName: string]: any;
  5. }
函数类型

可以使用接口表示函数类型,需要给接口定义一个调用签名:

</>复制代码

  1. interface SearchFunc {
  2. (source: string, subString: string): boolean;
  3. }
  4. let mySearch: SearchFunc;
  5. mySearch = function(source: string, subString: string) {
  6. const result = source.indexOf(subString);
  7. return result > -1;
  8. };

和变量类型的接口不同,函数的参数名不需要与接口定义的名字相匹配。函数的参数会逐个进行检查,要求对应位置上的参数类型是兼容的:

</>复制代码

  1. let mySearch: SearchFunc;
  2. mySearch = function(src: string, sub: string): boolean {
  3. let result = src.indexOf(sub);
  4. return result > -1;
  5. };

如果没有指定类型,TypeScript 的类型系统会推断出参数类型,因为函数直接赋值给了 SearchFunc 类型变量:

</>复制代码

  1. let mySearch: SearchFunc;
  2. mySearch = function(src, sub) {
  3. let result = src.indexOf(sub);
  4. return result > -1;
  5. };
可索引的类型

可索引类型具有一个 索引签名,它描述了对象索引的类型,还有相应的索引返回值类型。

</>复制代码

  1. interface StringArray {
  2. [index: number]: string;
  3. }
  4. let myArray: StringArray;
  5. myArray = ["Bob", "Fred"];
  6. let myStr: string = myArray[0];

上面的例子,定义了 StringArray 接口,它具有索引签名。这个索引签名表示了当用 number 去索引 StringArray 时会得到 string 类型的返回值。

TypeScript 支持两种索引签名:字符串和数字。可以同时使用两种类型的索引,但是数字索引的返回值必须是字符串索引返回值类型的子类型。其实,这相当于 JavaScript 对象的 key,数字类型会转换成字符串类型再取值。

字符串索引签名能够很好的描述 dictionary 模式,并且也会确保所有属性与其返回值类型相匹配:

</>复制代码

  1. interface NumberDictionary {
  2. [index: string]: number;
  3. length: number; // 可以,length 是 number 类型
  4. name: string; // 错误,name 的类型与索引类型返回值的类型不匹配
  5. }

可以将索引签名设置为只读,这样就防止了给索引赋值:

</>复制代码

  1. interface ReadonlyStringArray {
  2. readonly [index: number]: string;
  3. }
  4. let myArray: ReadonlyStringArray = ["Alice", "Bob"];
  5. myArray[2] = "Mallory"; // error!
类类型

类可以使用 implements 关键字实现某一个接口:

</>复制代码

  1. interface Point {
  2. x: number;
  3. y?: number;
  4. getX(): number;
  5. }
  6. class MyPoint implements Point {
  7. constructor(x: number) {
  8. this.x = x;
  9. }
  10. x: number;
  11. getX(): number {
  12. return this.x;
  13. }
  14. }

接口描述了类的公共部分,而不是公共和私有两部分。它不会帮你检查类是否具有某些私有成员。

继承接口

接口之间可以相互继承,并且一个接口还可以继承多个接口,就是将一个接口里的成员复制到另一个接口中:

</>复制代码

  1. interface Shape {
  2. color: string;
  3. }
  4. interface PenStroke {
  5. penWidth: number;
  6. }
  7. interface Square extends Shape, PenStroke {
  8. sideLength: number;
  9. }
  10. let square = {};
  11. square.color = "blue";
  12. square.sideLength = 10;
  13. square.penWidth = 5.0;
混合类型

一个对象可以同时具有多种类型。一个例子就是,一个对象可以同时做为函数和对象使用,并带有额外的属性:

</>复制代码

  1. interface Counter {
  2. (start: number): string;
  3. interval: number;
  4. reset(): void;
  5. }
  6. function getCounter(): Counter {
  7. let counter = function(start: number) {};
  8. counter.interval = 123;
  9. counter.reset = function() {};
  10. return counter;
  11. }
  12. let c = getCounter();
  13. c(10);
  14. c.reset();
  15. c.interval = 5.0;

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/106847.html

相关文章

  • TypeScript 初识 - 类

    摘要:抽象类抽象类做为其它字类的基类使用,一般不会直接被实例化。抽象类中可以包含具体实现,接口不能。抽象类在运行时是可见的,可以通过判断。接口只能描述类的公共部分,不会检查私有成员,而抽象类没有这样的限制。 一个普通的类 class Greeter { greeting: string; constructor(message: string) { this....

    邹强 评论0 收藏0
  • TypeScript 初识

    摘要:当你陷在一个中大型项目中时应用日趋成为常态,没有类型约束类型推断,总有种牵一发而动全身的危机和束缚。总体而言,这些付出相对于代码的健壮性和可维护性,都是值得的。目前主流的都为的开发提供了良好的支持,比如和。参考资料中文文档 文章博客地址:http://pinggod.com/2016/Typescript/ TypeScript 是 JavaScript 的超集,为 JavaScrip...

    iliyaku 评论0 收藏0
  • TypeScript 初识 - 内置对象

    摘要:自带的内置对象都可以直接在中当作定义好的类型。的内置对象标准提供了常用的内置对象等。在不需要额外引入就可以直接使用这些内置对象用写不是内置对象的一部分,想要写时提示,需要引入第三方声明文件 JavaScript 自带的内置对象都可以直接在 TypeScript 中当作定义好的类型。 TypeScript 核心库的定义文件 TypeScript 核心库的定义文件定义了所有浏览器环境需要用...

    lordharrd 评论0 收藏0
  • TypeScript 初识 - 基础

    摘要:安装通过命令进行安装创建文件文件名以扩展名结尾编译通过命令行进行编译通过运行任务的方式进行编译下使用显示运行任务,选择构建进行编译。 安装 通过 npm 命令进行安装: $ npm i -g typescript 创建 ts 文件 Typescript 文件名以 ts 扩展名结尾: function hello(value: string) { console.log(`hel...

    CODING 评论0 收藏0
  • TypeScript 初识 - 函数

    摘要:函数类型函数返回值类型我们可以给每个参数添加类型之后再为函数本身添加返回值类型。能够根据返回语句自动推断出返回值类型,所以通常可以省略它匿名函数完整函数类型完整的函数类型包含两部分参数类型和返回值类型。 函数是 JavaScript 的第一等公民,函数在 JavaScript 中可以实现抽象层、模拟类、信息隐藏和模块等等。TypeScript 在一定基础上扩展了函数的功能。 函数类型 ...

    ARGUS 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<