导航


HTML

CSS

JavaScript

浏览器 & 网络

版本管理

框架

构建工具

TypeScript

性能优化

软实力

算法

UI、组件库

Node

冷门技能

1. 对「对象」的形状进行描述

// 基本使用
interface IPerson {
  name: string;
  age: number;
}
const per: IPerson = { name: 'Lance', age: 23 };
// 对象的实现得和接口完全匹配
interface IPerson { // 接口一旦定义,per 对象的属性就必须得只有两个属性,不能多也不能少
  name: string;
  age: number;
}
const per: IPerson = { name: 'Lance' }; // 不写 age ,会报错缺少 age 属性
// 可选属性
interface IPerson {
  name: string;
  age: number;
  weight?: number; // 可选属性
}
const per: IPerson = { name: 'Lance', age: 23 }; // 可以不写 weight
// 让对象的实现有 interface 中没有的属性(任意属性)
interface IPerson {
  name: string;
  age: number;
  weight?: number; // 可选属性
  [propName: string]: any // 对象可以有其他属性
}
const per: IPerson = { name: 'Lance', age: 23, run: function() {}, sex: 'male' }
// 只读属性
interface IPerson {
  readonly id: number;
  name: string;
  age: number;
  weight?: number; // 可选属性
  [propName: string]: any // 对象可以有其他属性
}
const per: IPerson = { id: 1, name: 'Lance', age: 23, sex: 'male' }
per.id = 233; // 报错

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

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

定义函数类型接口

接口也可以描述函数类型。因为函数也可以理解为一个对象

interface SearchFunc {
  (source: string, subString: string): boolean;
}

let mySearch: SearchFunc;
mySearch = function(source: string, subString: string) {
  let result = source.search(subString);
  return result > -1;
}

定义可索引的类型接口(数组)

interface StringArray {
  [index: number]: string;
}

let myArray: StringArray;
myArray = ["Bob", "Fred"];

let myStr: string = myArray[0];

Untitled

2. 对类的一部分行为的抽象

定义类类型接口

interface ClockInterface {
    currentTime: Date;
}

// 表示 Clock 类要 实现 ClockInterface 接口
class Clock implements ClockInterface {
    currentTime: Date;
    constructor(h: number, m: number) { }
}
interface ClockInterface {
    currentTime: Date;
    setTime(d: Date);
}

class Clock implements ClockInterface {
    currentTime: Date;
    setTime(d: Date) {
        this.currentTime = d;
    }
    constructor(h: number, m: number) { }
}
interface Alarm {
  alert(): void;
}
interface Light {
  color: string;
}
class Door {}
class SecurityDoor extends Door implements Alarm {
  alert() {}
}
class Car implements Alarm, Light {
  color = 'red';
  alert() {}
}

类静态部分与实例部分的区别

Untitled

interface ClockConstructor {
    new (hour: number, minute: number): ClockInterface;
}
interface ClockInterface {
    tick();
}

// 通过函数来检查类的构造器
function createClock(ctor: ClockConstructor, hour: number, minute: number): ClockInterface {
    return new ctor(hour, minute);
}

class DigitalClock implements ClockInterface {
    constructor(h: number, m: number) { }
    tick() {
        console.log("beep beep");
    }
}
class AnalogClock implements ClockInterface {
    constructor(h: number, m: number) { }
    tick() {
        console.log("tick tock");
    }
}

let digital = createClock(DigitalClock, 12, 17);
let analog = createClock(AnalogClock, 7, 32);

接口继承接口

interface Shape {
    color: string;
}

interface Square extends Shape {
    sideLength: number;
}

let square = <Square>{}; // <Square>{} 是类型的断言 语法:<类型>值
square.color = "blue";
square.sideLength = 10;
interface Shape {
    color: string;
}

interface PenStroke {
    penWidth: number;
}

interface Square extends Shape, PenStroke {
    sideLength: number;
}

let square = <Square>{};
square.color = "blue";
square.sideLength = 10;
square.penWidth = 5.0;

混合类型

先前我们提过,接口能够描述JavaScript里丰富的类型。 因为JavaScript其动态灵活的特点,有时你会希望一个对象可以同时具有上面提到的多种类型。 一个例子就是,一个对象可以同时做为函数和对象使用,并带有额外的属性。

interface Counter {
    (start: number): string; // 定义函数的接口
    interval: number; // 这个函数又有自己的方法(把函数当做对象)
    reset(): void;
}

function getCounter(): Counter {
    let counter = <Counter>function (start: number) { };
    counter.interval = 123;
    counter.reset = function () { };
    return counter;
}

let c = getCounter();
c(10);
c.reset();
c.interval = 5.0;

接口继承类

class Control {
    private state: any;
}

interface SelectableControl extends Control {
    select(): void;
}

class Button extends Control implements SelectableControl {
    select() { }
}

class TextBox extends Control {
    select() { }
}

// 错误:“Image”类型缺少“state”属性。
class Image implements SelectableControl {
  // Image 没法实现 SelectableControl 因为 Control 中的 state 是私有的,没法实现
    select() { }
}

class Location {

}