TypeScript定义接口

要想掌握typescript的知识,接口是其必经之路。很多东西都需要接触到接口,接口除了对类的一部分行为进行抽象以外,也常用于对对象的形状进行描述。接下来我们就一起来学习一下,如何才能熟练掌握接口的使用。

一. 为什么要使用接口

1.1. JavaScript存在的问题

我们在JavaScript中定义一个函数,用于获取一个用户的姓名和年龄的字符串:

  1. const getUserInfo = function(user) {
  2.   return`name: ${user.name}, age: ${user.age}`
  3. }

正确的调用方法应该是下面的方式:

  1. getUserInfo({name: "coderwhy", age: 18})

但是当项目比较大,或者多人开发时,会出现错误的调用方法:

  1. // 错误的调用
  2. getUserInfo() // Uncaught TypeError: Cannot read property 'name' of undefined
  3. console.log(getUserInfo({name: "coderwhy"})) // name: coderwhy, age: undefined
  4. getUserInfo({name: "codewhy", height: 1.88}) // name: coderwhy, age: undefined

因为JavaScript是弱类型的语言,所以并不会对我们传入的代码进行任何的检测,但是在之前的javaScript中确确实实会存在很多类似的安全隐患。

如何避免这样的问题呢?

  • 当然是使用TypeScript来对代码进行重构

1.2. TypeScript代码重构一

我们可以使用TypeScript来对上面的代码进行改进:

  1. const getUserInfo = (user: {name: string, age: number}): string => {
  2.   return`name: ${user.name} age: ${user.age}`;
  3. };

正确的调用是如下的方式:

  1. getUserInfo({name: "coderwhy", age: 18});

如果调用者出现了错误的调用,那么TypeScript会直接给出错误的提示信息:

  1. // 错误的调用
  2. getUserInfo(); // 错误信息:An argument for 'user' was not provided.
  3. getUserInfo({name: "coderwhy"}); // 错误信息:Property 'age' is missing in type '{ name: string; }'
  4. getUserInfo({name: "coderwhy", height: 1.88}); // 错误信息:类型不匹配

这样确实可以防止出现错误的调用,但是我们在定义函数的时候,参数的类型和函数的类型都是非常长的,代码非常不便于阅读

所以,我们可以使用接口来对代码再次进行重构。

1.3. TypeScript代码重构二

现在我们使用接口来对user的类型进行重构。

接口重构一:参数类型使用接口定义

我们先定义一个IUser接口:

  1. // 先定义一个接口
  2. interface IUser {
  3.   name: string;
  4.   age: number;
  5. }

接下来我们看一下函数如何来写:

  1. const getUserInfo = (user: IUser): string => {
  2.   return`name: ${user.name}, age: ${user.age}`;
  3. };
  4. // 正确的调用
  5. getUserInfo({name: "coderwhy", age: 18});
  6. // 错误的调用,其他也是一样
  7. getUserInfo();

接口重构二:函数的类型使用接口定义好(后面会详细讲解接口函数的定义)

我们先定义两个接口:

  • 第二个接口定义有一个警告,我们暂时忽略它,它的目的是如果一个函数接口只有一个方法,那么可以使用type来定义
  • type IUserInfoFunc = (user: IUser) => string;
  1. interface IUser {
  2.   name: string;
  3.   age: number;
  4. }
  5. interface IUserInfoFunc {
  6.   (user: IUser): string;
  7. }

接着我们去定义函数和调用函数即可:

  1. const getUserInfo: IUserInfoFunc = (user) => {
  2.   return`name: ${user.name}, age: ${user.age}`;
  3. };
  4. // 正确的调用
  5. getUserInfo({name: "coderwhy", age: 18});
  6. // 错误的调用
  7. getUserInfo();

二. 接口的基本使用

2.1. 接口的定义方式

和其他很多的语言类似,TypeScript中定义接口也是使用interface关键字来定义:

  1. interface IPerson {
  2.   name: string;
  3. }

你会发现我都在接口的前面加了一个I,这是tslint要求的,否则会报一个警告

  • 要不要加前缀是根据公司规范和个人习惯
  1. interface name must start with a capitalized I

当然我们可以在tslint中关闭掉它:在rules中添加如下规则

  1. "interface-name" : [true, "never-prefix"]

2.2. 接口中定义方法

定义接口中不仅仅可以有属性,也可以有方法:

  1. interface Person {
  2.   name: string;
  3.   run(): void;
  4.   eat(): void;
  5. }

如果我们有一个对象是该接口类型,那么必须包含对应的属性和方法:

  1. const p: Person = {
  2.   name: "why",
  3.   run() {
  4.     console.log("running");
  5.   },
  6.   eat() {
  7.     console.log("eating");
  8.   },
  9. };

2.3. 可选属性的定义

默认情况下一个变量(对象)是对应的接口类型,那么这个变量(对象)必须实现接口中所有的属性和方法。

但是,开发中为了让接口更加的灵活,某些属性我们可能希望设计成可选的(想实现可以实现,不想实现也没有关系),这个时候就可以使用可选属性(后面详细讲解函数时,也会讲到函数中有可选参数):

  1. interface Person {
  2.   name: string;
  3.   age?: number;
  4.   run(): void;
  5.   eat(): void;
  6.   study?(): void;
  7. }

上面的代码中,我们增加了age属性和study方法,这两个都是可选的:

  • 可选属性如果没有赋值,那么获取到的值是undefined;
  • 对于可选方法,必须先进行判断,再调用,否则会报错;
  1. const p: Person = {
  2.   name: "why",
  3.   run() {
  4.     console.log("running");
  5.   },
  6.   eat() {
  7.     console.log("eating");
  8.   },
  9. };
  10. console.log(p.age); // undefined
  11. p.study(); // 不能调用可能是“未定义”的对象。

正确的调用方式如下:

  1. if (p.study) {
  2.   p.study();
  3. }

2.4. 只读属性的定义

默认情况下,接口中定义的属性可读可写:

  1. console.log(p.name);
  2. p.name = "流川枫";

如果一个属性,我们只是希望在定义的时候就定义值,之后不可以修改,那么可以在属性的前面加上一个关键字:readonly

  1. interface Person {
  2.   readonly name: string;
  3.   age?: number;
  4.   run(): void;
  5.   eat(): void;
  6.   study?(): void;
  7. }

当我在name前面加上readonly时,赋值语句就会报错:

  1. console.log(p.name);
  2. p.name = "流川枫"; // Cannot assign to 'name' because it is a read-only property.

三. 接口的高级使用

3.1. 函数类型的定义

接口不仅仅可以定义普通的对象类型,也可以定义函数的类型

  1. // 函数类型的定义
  2. interface SumFunc {
  3.   (num1: number, num2: number): number;
  4. }
  5. // 定义具体的函数
  6. const sum: SumFunc = (num1, num2) => {
  7.   return num1 + num2;
  8. };
  9. // 调用函数
  10. console.log(sum(20, 30));

不过上面的接口中只有一个函数,TypeScript会给我们一个建议,可以使用type来定义一个函数的类型:

  1. type SumFunc = (num1: number, num2: number) =>number;

关于type的更多用户,我们后面专门进行讲解,暂时不在接口中展开讨论。

3.2. 可索引类型的定义

和使用接口描述函数的类型差不多,我们也可以使用接口来描述 可索引类型

  • 比如一个变量可以这样访问:a[3],a["name"]

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

  1. // 定义可索引类型的接口
  2. interface RoleMap {
  3.   [index: number]: string;
  4. }
  5. // 赋值具体的值
  6. // 赋值方式一:
  7. const roleMap1: RoleMap = {
  8.   0: "学生",
  9.   1: "讲师",
  10.   2: "班主任",
  11. };
  12. // 赋值方式二:因为数组本身是可索引的值
  13. const roleMap2 = ["鲁班七号", "露娜", "李白"];
  14. // 取出对应的值
  15. console.log(roleMap1[0]); // 学生
  16. console.log(roleMap2[1]); // 露娜

上面的案例中,我们的索引签名是数字类型,TypeScript支持两种索引签名:字符串和数字

我们来定义一个字符串的索引类型:

  1. interface RoleMap {
  2.   [name: string]: string;
  3. }
  4. const roleMap: RoleMap = {
  5.   aaa: "鲁班七号",
  6.   bbb: "露娜",
  7.   ccc: "李白",
  8. };
  9. console.log(roleMap.aaa);
  10. console.log(roleMap["aaa"]); // 警告:不推荐这样来取

可以同时使用两种类型的索引,但是数字索引的返回值必须是字符串索引返回值类型的子类型:

  • 这是因为当使用 number来索引时,JavaScript会将它转换成string然后再去索引对象。
  1. class Person {
  2.   private name: string = "";
  3. }
  4. class Student extends Person {
  5.   private sno: number = 0;
  6. }
  7. // 下面的代码会报错
  8. interface IndexSubject {
  9.   [index: number]: Person;
  10.   [name: string]: Student;
  11. }

代码会报如下错误:

  1. 数字索引类型“Person”不能赋给字符串索引类型“Student”。

修改为如下代码就可以了:

  1. interface IndexSubject {
  2.   [index: number]: Student;
  3.   [name: string]: Person;
  4. }

下面的代码也会报错:

  • letter索引得到结果的类型,必须是Person类型或者它的子类型
  1. interface IndexSubject {
  2.   [index: number]: Student;
  3.   [name: string]: Person;
  4.   letter: string;
  5. }

3.3. 接口的实现

注意:在这个小节以及下一个小节中,我们会写一些类,但是目前还没有详细学习类的语法(虽然TS的类和ES6的非常相似)。

大家可以先知道我们的类如何定义,如何去和接口配合使用的即可,一些细节我会有专门的文章来解决类的使用。

接口除了定义某种类型规范之后,也可以和其他编程语言一样,让一个类去实现某个接口,那么这个类就必须明确去拥有这个接口中的属性和实现其方法:

  • 下面的代码中会有关于修饰符的警告,暂时忽略,后面详细讲解
  1. // 定义一个实体接口
  2. interface Entity {
  3.   title: string;
  4.   log(): void;
  5. }
  6. // 实现这样一个接口
  7. class Post implements Entity {
  8.   title: string;
  9.   constructor(title: string) {
  10.     this.title = title;
  11.   }
  12.   log(): void {
  13.     console.log(this.title);
  14.   }
  15. }

思考:我定义了一个接口,但是我在继承这个接口的类中还要写接口的实现方法,那我不如直接就在这个类中写实现方法岂不是更便捷,还省去了定义接口?这是一个初学者经常会有疑惑的地方。

从思考方式上,为什么需要接口?

我们从生活出发理解接口

比如你去三亚/杭州旅游, 玩了一上午后饥饿难耐, 你放眼望去,会注意什么? 饭店!!

你可能并不会太在意这家饭店叫什么名字, 但是你知道只要后面有饭店两个字, 就意味着这个地方必然有饭店的实现 – 做各种菜给你吃;

接口就好比饭店/酒店/棋牌室这些名词后面添加的附属词, 当我们看到这些附属词后就知道它们具备的功能

从代码设计上,为什么需要接口?

  • 在代码设计中,接口是一种规范;
  • 接口通常用于来定义某种规范, 类似于你必须遵守的协议, 有些语言直接就叫protocol;
  • 站在程序角度上说接口只规定了类里必须提供的属性和方法,从而分离了规范和实现,增强了系统的可拓展性和可维护性;

当然,对于初次接触接口的人,还是很难理解它在实际的代码设计中的好处,这点慢慢体会,不用心急。

3.3. 接口的继承

和类相似(后面我们再详细学习类的知识),接口也是可以继承接口来提供复用性:

  • 注意:继承使用extends关键字
  1. interface Barkable {
  2.   barking(): void;
  3. }
  4. interface Shakable {
  5.   shaking(): void;
  6. }
  7. interface Petable extends Barkable, Shakable {
  8.   eating(): void;
  9. }

接口Petable继承自Barkable和Shakable,另外我们发现一个接口可以同时继承自多个接口

如果现在有一个类实现了Petable接口,那么不仅仅需要实现Petable的方法,也需要实现Petable继承自的接口中的方法:

  • 注意:实现接口使用implements关键字
  1. class Dog implements Petable {
  2.   barking(): void {
  3.     console.log("汪汪叫");
  4.   }
  5.   shaking(): void {
  6.     console.log("摇尾巴");
  7.   }
  8.   eating(): void {
  9.     console.log("吃骨头");
  10.   }
  11. } 

如果你觉得接口的内容就仅仅局限于此,那可就大错特错了,接口也要结合其他的知识同时运用,这其中必然少不了你反复的练习,如果你想提升你的编程能力,那就关注我,我会为你发布更多的精彩教程,帮助你突破瓶颈,提升自我。

TypeScript之路----探索接口(interface)的奥秘的更多相关文章

  1. TypeScript学习指南第二章--接口(Interface)

    接口(Interface) TypeScript的核心机制之一在于它的类型检查系统(type-checker)只关注一个变量的"模型(shape)" 稍后我们去了解这个所谓的形状是 ...

  2. typescript接口---interface

    假如我现在需要批量生产一批对象,这些对象有相同的属性,并且对应属性值的数据类型一致.该怎么去做? 在ts中,因为要检验数据类型,所以必须对每个变量进行规范,自然也提供了一种批量规范的功能.这个功能就是 ...

  3. delphi 接口Interface

    学习 delphi 接口 一切都是纸老虎!!! 第四章          接口 前不久,有位搞软件的朋友给我出了个谜语.谜面是“相亲”,让我猜一软件术语.我大约想了一分钟,猜 出谜底是“面向对象”.我 ...

  4. 使用Typescript重构axios(十一)——接口扩展

    0. 系列文章 1.使用Typescript重构axios(一)--写在最前面 2.使用Typescript重构axios(二)--项目起手,跑通流程 3.使用Typescript重构axios(三) ...

  5. TypeScript 真香系列——接口篇

    接口带来了什么好处 好处One —— 过去我们写 JavaScript JavaScript 中定义一个函数,用来获取一个用户的姓名和年龄的字符串: const getUserInfo = funct ...

  6. TypeScript的泛型接口 泛型类接口

    /* typeScript中的泛型 泛型接口 */ //函数类型接口 /* interface ConfigFn{ (value1:string,value2:string):string; } va ...

  7. TypeScript(4)接口

    介绍 TypeScript 的核心原则之一是对值所具有的结构进行类型检查.我们使用接口(Interfaces)来定义对象的类型.接口是对象的状态(属性)和行为(方法)的抽象(描述) 接口初探 声明接口 ...

  8. java中的接口interface

    关于接口 接口描述了实现了它的类拥有什么功能.因为Java是强类型的,所以有些操作必须用接口去约束和标记.接口作为类的能力的证明,它表明了实现了接口的类能做什么. 类似与class,interface ...

  9. php中的抽象类(abstract class)和接口(interface)

    一. 抽象类abstract class 1 .抽象类是指在 class 前加了 abstract 关键字且存在抽象方法(在类方法 function 关键字前加了 abstract 关键字)的类. 2 ...

随机推荐

  1. LibreOJ #526. 「LibreOJ β Round #4」子集

    二次联通门 : LibreOJ #526. 「LibreOJ β Round #4」子集 /* LibreOJ #526. 「LibreOJ β Round #4」子集 考虑一下,若两个数奇偶性相同 ...

  2. Three.js中的div标签跟随(模型弹框)

    目录 Three.js中的div标签跟随(模型弹框) 参考官方案例 核心渲染器 用法 注意事项 Three.js中的div标签跟随(模型弹框) 参考官方案例 核心渲染器 three.js-master ...

  3. Ubuntu16.04忘记MySQL5.7的root用户密码之解决方案

    其实也就四步,如下: 修改配置文件 sudo vimi /etc/mysql/mysql.conf.d/mysqld.cnf 并在 在[mysqld]下方的skip-external-locking下 ...

  4. 【纸模】六角大王 Super 5.6 CHS 简体中文版 U20080725+[手册]窗口与工具的概要(PDF格式)

    六角大王5.6简体中文版中文化:star21 主界面<ignore_js_op> 人体生成模式<ignore_js_op> 动画<ignore_js_op> < ...

  5. 屏幕录制 -- web前端

    前端使用html5.ffmpeg实现录屏摄像等功能 https://tong-h.github.io/2018/11/06/streamcapture/ JSCapture – 基于 HTML5 实现 ...

  6. 迁移mysql数据位置

    查看位置: show variables like '%datadir%'; /var/lib/mysql

  7. Spring Boot核心原理

    Spring Boot核心原理 spring-boot-starter-xxx  方便开发和配置 1.没有depoy setup tomcat 2.xml文件里面的没有没有了 @SpringBootA ...

  8. centos sqlite3安装及简单命令

    安装:方法一:wget http://www.sqlite.org/sqlite-autoconf-3070500.tar.gztar xvzf sqlite-autoconf-3070500.tar ...

  9. leetcode 611. Valid Triangle Number 、259. 3Sum Smaller(lintcode 918. 3Sum Smaller)

    这两个题几乎一样,只是说611. Valid Triangle Number满足大于条件,259. 3Sum Smaller满足小于条件,两者都是先排序,然后用双指针的方式. 611. Valid T ...

  10. cordova run android 可能遇到的错误解决

    运行: ionic cordova build 等待下载,然后根据提示 输入android或者ios平台,即可 运行cordova run android 报错: 最快捷的解决方法就是使用Androi ...