likes
comments
collection
share

TypeScript 中 type 和 interface 的区别

作者站长头像
站长
· 阅读数 41

1、interface

interface(接口) 是 TS 设计出来用于定义对象类型的,可以对对象的形状进行描述。

// 定义
interface Person {
  name: string
  age: number
}

// 使用
const person: Person = {
  name: 'lin',
  age: 18,
};

2、type

type (类型别名),顾名思义,类型别名只是给类型起一个新名字。

它并不是一个类型,只是一个别名而已

// 定义
type Person = {
  name: string;
  age: number;
}

// 使用
const person: Person = {
  name: 'lin',
  age: 18,
};

3、两者相同点

3.1、都可以定义一个对象或函数

定义对象上面已经有例子了,我们来看一下如何定义函数。

// 使用 type 定义
type addType = (num1: number, num2: number) => number;

// 使用 interface 定义
interface addType {
  (num1: number, num2: number): number;
}

// 上面这两种写法都可以定义函数类型,效果是一样的

// 使用
const add: addType = (num1, num2) => {
  return num1 + num2;
};

3.2、都允许继承 extends(也可以叫拓展)

  • interface 可以使用 extends 关键字来扩展其他接口,实现接口的继承。
  • type 使用交叉类型(&)来合并多个类型,实现类型的组合。

我们定义一个 Person 类型和 Student 类型,Student 继承自 Person,可以有下面四种方式:

interface 继承 interface

interface Person { 
  name: string;
}

interface Student extends Person { 
  grade: number;
}

// 使用
const person: Student = {
  name: 'lin',
  grade: 100,
};

type 继承 type

type Person = { 
  name: string;
}
type Student = Person & { grade: number }; // 用交叉类型

interface 继承 type

type Person = {
  name: string;
};

interface Student extends Person {
  grade: number;
}

type 继承 interface

interface Person {
  name: string;
}

type Student = Person & { grade: number }; // 用交叉类型

interface 使用 extends 实现继承, type 使用交叉类型实现继承。

注意:interface 不能继承一个联合类型

type Input = {
  a: string;
};
type Output = {
  b: string;
};

type InputOrOutput = Input | Output;

// 报错:接口只能扩展使用静态已知成员的对象类型或对象类型的交集。
interface C extends InputOrOutput {
  c: string;
}

// 可以用type实现
type C = InputOrOutput & { c: string };

// 使用
const value: C = {
  a: '1',
  b: '2',
  c: '3',
};

4、两者不同点

4.1、interface 可以,type 不行

4.1.1、合并重复声明

interface Person {
  name: string;
}

interface Person {
  // 重复声明 interface,就合并了
  age: number;
}

const person: Person = {
  name: 'lin',
  age: 18,
};

重复声明 type ,就报错了

type Person = {
  name: string;
};

type Person = {
  // Duplicate identifier 'Person'
  age: number;
};

TypeScript 中 type 和 interface 的区别

4.1.2、定义类和抽象类的实现

如果需要定义一个类或抽象类的实现,通常应该使用 interface 而不是 typeinterface 可以被类实现,而 type 不能。

// 使用 interface 定义抽象类的实现
interface Shape {
  area(): number;
}

abstract class Circle implements Shape {
  abstract radius: number;

  area() {
    return Math.PI * this.radius ** 2;
  }
}

4.2、type 可以,interface 不行

类型别名会给一个类型起个新名字。 类型别名有时和接口很像,但是可以作用于原始值,联合类型,元组以及其它任何你需要手写的类型。

4.2.1、声明基本类型、联合类型、交叉类型、元组

// 基本类型
type Name = string; 

type Person = {
  name: Name;
};


// 联合类型
type arrItem = number | string; 

const arr: arrItem[] = [1, '2', 3];


// 交叉类型
type Student = Person & { grade: number }; 

type Teacher = Person & { major: string };


// 元组类型
type StudentAndTeacherList = [Student, Teacher]; 

const list: StudentAndTeacherList = [
  { name: 'lin', grade: 100 },
  { name: 'liu', major: 'Chinese' },
];

4.2.2、定义映射类型

使用 type 可以定义映射类型,根据现有类型创建新类型。

type ReadOnly<T> = {
  readonly [K in keyof T]: T[K];
};

type User = {
  name: string;
  age: number;
};

type ReadOnlyUser = ReadOnly<User>;

4.2.3、条件类型

使用 type 可以定义条件类型,根据不同条件返回不同的类型。

type NonNullable<T> = T extends null | undefined ? never : T;

type MaybeString = string | null | undefined;
type NonNullMaybeString = NonNullable<MaybeString>;

4.2.4、模板字符串字面量类型

使用 type 可以定义模板字符串字面量类型,用于字符串字面量的静态检查。

type Greeting = `Hello, ${string}!`;

// 合法
const greeting: Greeting = "Hello, World!"; 

// 不合法
const invalidGreeting: Greeting = 42; // 报错:不能将类型 42 分配给类型“`Hello, ${string}!`”。

4.2.5、字符串字面量类型和数字字面量类型

使用 type 可以定义字符串字面量类型和数字字面量类型。

type Gender = "male" | "female";
type DayOfWeek = 0 | 1 | 2 | 3 | 4 | 5 | 6;

5、总结

interfacetype 有着相似的功能,但是它们也有各自独有的特点。我们可以把它们区分为以下几点:

5.1、interface适用于

  • 用于声明对象类型的接口
  • 需要继承或声明合并的接口
  • 声明函数类型的接口
  • 需要在类型注释中使用类型的接口

5.2、type适用于

  • 声明原始类型的别名
  • 定义复杂的类型,如交叉类型、联合类型、元组和映射类型
  • 使用泛型的类型别名

通常情况下,当我们主要处理对象时,我们更倾向于使用 interface。而当我们需要更加高级的类型定义时,我们更倾向于使用 type

如果不清楚什么时候用 interface/type,能用 interface 实现,就用 interface。如果不能就用 type 实现。

转载自:https://juejin.cn/post/7274856810669588516
评论
请登录