在 TypeScript 中扩展多个接口

使用 extends 关键字扩展 TypeScript 中的多个接口,例如 interface Developer extends Person, Employee {}。 extends 关键字允许我们从指定类型复制成员并将新成员添加到最终接口。

interface Person {
  name: string;
}

interface Employee {
  id: number;
  salary: number;
}

interface Developer extends Person, Employee {
  language: string;
}

const dev: Developer = {
  id: 1,
  name: 'Tom',
  salary: 100,
  language: 'TypeScript',
};

extends 关键字消除了必须在多个位置重复其他类型的成员的需要,并向代码阅读者显示接口之间的关系。

我们可以根据需要扩展多个接口,方法是用逗号分隔接口。

我们不需要向最终接口添加任何新成员,并且可以使用 extends 关键字来简单地组合接口。

nterface Person {
  name: string;
}

interface Employee {
  id: number;
  salary: number;
}

// ?️ Combine the Person and Employee interfaces
interface Developer extends Person, Employee {}

// ?️ Alternatively use intersection type
type Developer2 = Person & Employee;

const dev: Developer = {
  id: 1,
  name: 'Tom',
  salary: 100,
};

上面的示例显示了如何组合两个或多个接口而不向它们添加新属性。

如果我们需要从您扩展的接口之一覆盖一个属性,我们可以使用 Omit 实用程序类型。

interface Person {
  name: string;
}

interface Employee {
  id: number;
  salary: number;
}

// for multiple use - Omit<Employee, 'id' | 'salary'>
interface Developer extends Person, Omit<Employee, 'id'> {
  id: string; // ?️ override type of id
}

const dev: Developer = {
  id: 'dev-1',
  name: 'Tom',
  salary: 100,
};

Omit 实用程序类型通过从提供的类型中选择属性并删除指定的键来构造一个新类型。

在示例中,我们从 Employee 类型扩展时删除了 id 属性,因此我们可以覆盖它的类型。

如果我们没有删除该属性并尝试在 Developer 接口中指定不同的类型,我们会得到一个错误。

interface Person {
  name: string;
}

interface Employee {
  id: number;
  salary: number;
}

// ⛔️ Error: Interface 'Developer' incorrectly extends interface 'Employee'.
// Types of property 'id' are incompatible.
//   Type 'string' is not assignable to type 'number'.ts(2430)
interface Developer extends Person, Employee {
  id: string; // ?️ override type of id
}

在 TypeScript 中扩展多个接口

TypeScript 告诉我们,我们不能从接口扩展并使用具有不同、不兼容类型的相同属性名称。

解决此问题的最佳方法是在从接口扩展时删除类型并在新接口中覆盖它。

扩展时,我们可以使用字符串文字的联合来省略类型中的多个属性。

interface Person {
  name: string;
}

interface Employee {
  id: number;
  salary: number;
  tasks: string[];
}

interface Developer extends Person, Omit<Employee, 'id' | 'salary'> {
  id: string; // ?️ override type of property
  salary: string; // ?️ override type of property
  language: string;
}

const dev: Developer = {
  id: 'dev-1',
  name: 'Tom',
  salary: '100 K',
  language: 'TypeScript',
  tasks: ['develop', 'test'],
};

要将多个属性传递给 Omit 实用程序类型,请使用竖线 | 分隔属性。

扩展时可以使用相同的方法从类型中删除某些属性 – 省略属性并且不要重新定义它们。

请注意,从接口扩展时,我们可以为相同的属性名称提供更窄的类型。

interface Person {
  name: string;
}

interface Employee {
  id: number | string; // ?️ number OR string (wide)
  salary: number;
}

interface Developer extends Person, Employee {
  id: string; // ?️ only string (narrow)
  language: string;
}

const dev: Developer = {
  id: 'dev-50', // ?️ can only be string now
  name: 'Tom',
  salary: 100,
  language: 'TypeScript',
};

Employee 接口中的 id 属性具有数字或字符串的类型,但是从接口扩展时,我们指定了更窄的字符串类型。

Developer 接口中的 id 属性只能是字符串类型。

允许用更具体的属性覆盖属性类型,但是,我们不能提供不兼容的类型。 如果我们需要这样做,则必须使用 Omit 实用程序类型来删除特定属性并覆盖其类型。

扩展接口的主要好处是:

  1. 减少重复,因为我们不必在接口之间复制属性。
  2. 向我们代码的读者发出信号,表明这两种类型之间存在关系。