TypeScript学习

基础类型

布尔

  • let is:boolean = false

数字

  • let n:number = 6
  • let n:number = 0xf00d
  • let n:number = 0b1010
  • let n:number = 0o744

字符串

  • let s:string = "fuck"
  • let s:string = `模板字符串`

数组

  • let list:number[] = [1,2,3]
  • let list:Array<number> = [1,2,3]

元祖 Tuple

  • let list:[string,number] = ["s",1]
  • 顺序也必须相同 , 越界赋值为联合类型

枚举

enum Color {Red,Green,Blue} //默认0开始
let c:Color = Color.Green   //值为1

//可赋值
enum Color {Red=1,Green=2,Blue=4} 
//可以获取其名字
let c:string = Color[1] //Red

Any(任意类型)

let notSure: any = 4
notSure = "maybe a string instead"
notSure = false

与Object类型的区别 , Object类型变量不允许调用上面的任意方法 , 但是any类型可以.

let notSure: any = 4
notSure.toFixed() // ok

let prettySure: Object = 4
prettySure.toFixed() // Error: Property 'toFixed' doesn't exist on type 'Object'.

在数组时 , any类型也有用

let list: any[] = [1, true, "free"];

Void

  • let unusable:void = undefined
  • 官网说可以赋值null,但是我测试会报错.

Null和Undefined

  • let u: undefined = undefined
  • let n: null = null

Never

永远不会返回值的类型

// 返回never的函数必须存在无法达到的终点
function error(message: string): never {
    throw new Error(message);
}

// 推断的返回值类型为never
function fail():never {
    return error("Something failed");
}

// 返回never的函数必须存在无法达到的终点
function infiniteLoop(): never {
    while (true) {
    }
}    

Object

除了number,string,boolean,symbol,null或undefined之外的类型。

  • let o: object = {}
  • let o: object = new String()
  • let o: object = ""//报错
  • let o: object = 1 //报错

类型断言

  • 尖括号 语法
let someValue: any = "this is a string";

let strLength: number = (<string>someValue).length;
  • as 语法
let someValue: any = "this is a string";

let strLength: number = (someValue as string).length;

两种等价 , 在用jsx时只有as语法可以用








变量声明

let

  • let h = "hello"
  • 声明会产生块级作用域
  • 不能在声明前访问 , 暂时性死区
    • 包括函数
      function foo() {
          // okay to capture 'a'
          return a;
      }
      
      // 不能在'a'被声明前调用'foo'
      // 运行时应该抛出错误
      foo();
      
      let a;
      
  • 不能多次声明

const

  • const numLivesForCat = 9
  • 与let相同作用域规则
  • 赋值后不可变








解构

结构数组

  • 取值
    • let [a, b] = [0, 1]
    • let [ , b , , d] = [0, 1, 2, 3] // b=1 d=3
  • 交换变量值
    • [a,b] = [b,a]
    • 数组内交换
      let a = [1, 2];
      [a[0], a[1]] = [a[1], a[0]]
      
  • 函数参数
    function f([first, second]: [number, number]) {
        console.log(first);
        console.log(second);
    }
    
    f([1, 2]);
    
  • ...创建剩余变量
    let [first, ...rest] = [1, 2, 3, 4];
    console.log(first); //  1
    console.log(rest);  //  [ 2, 3, 4 ]
    

对象解构

  • 取值
    • let {a, b} = {a: 1, b: 2}
    • ({a, b} = {a: 3, b: 4})//用括号包起来 , js会把{}解析为一个块语句而不是object
  • ...创建剩余变量
    let {a, ...c} = {a: 1, b: 2, c: 3};
    console.log(a); // 1
    console.log(c); // { b: 2, c: 3 }
    
  • 属性重命名
    let {a: a1, b: b1} = {a: 1, b: 2};
    console.log(a1);
    console.log(b1);
    
    若要指示类型

    let {a: a1, b: b1}: { a: number, b: number } = {a: 1, b: 2};
  • 默认值
    • 值为undefined时赋值默认值
    • let {a: a1, b: b1 = 3} = {a: 1};
  • 函数参数
    function f({a, b}: { a: number, b: number }): void {
        console.log(a);
        console.log(b);
    }
    f({a: 1, b: 2})   
    //加入参数默认值
    function f({a = 9, b = 9}: { a: number, b: number } = {a: 0, b: 0}): void {}
    
    对象解构请保持简单,否侧会难以理解.

展开

  • 数组展开
    let a = [0, ...[1, 2, 3], ...[4, 5, 6]] // [ 0, 1, 2, 3, 4, 5, 6 ]
    
  • 对象展开
    let a = {a: 2, ...{a: 1, b: 2}}
    console.log(a); // { a: 1, b: 2 }
    
    属性相同值会被后面的覆盖








接口

语法

interface LabelledValue {
  label: string
}

可选属性

  • 某些属性不一定是必须的,加?
    interface SquareConfig {
      color?: string
      width?: number
    }
    

只读属性 readonly

interface Point {
    //x,y的值无法被修改
    readonly x: number;
    readonly y: number;
} 
let p1: Point = { x: 10, y: 20 };
p1.x = 5; // error!

只读数组类型 ReadonlyArray

  • 此类型去掉了所有可变方法
    let r: ReadonlyArray<number> = [1, 2, 3, 4]
    r[0] = 1  // error
    r.push(1)  // error
    
  • 转换为普通数组
    let a: number[] = r as number[]
    

字符串索引签名

interface SquareConfig {
    color?: string;
    width?: number;
    [key: string]: any;
}
//可以添加任意属性
let a: SquareConfig = {test: 1}

函数类型

interface SearchFunc {
  //参数列表 : 返回值    
  (source: string, subString: string): boolean;
}
//参数名可以不同
let my: SearchFunc = function (src: string, subString: string): boolean {
    return src + subString === ""
}
//自动类型推导
let my: SearchFunc = function (src, subString){
    return src + subString === ""
}

可索引的类型

interface StringArray {
  //对象的 key类型:value类型  
  [index: number]: string;
}

let myArray: StringArray = ["Bob", ""];
  • ts支持两种签名:string和number
    interface NumberDictionary {
      [key: string]: number;
      length: number;    // 可以,length是number类型
      name: string       // 错误,`name`的类型与索引类型返回值的类型不匹配
    }
    
  • 只读索引签名
    interface ReadonlyStringArray {
        readonly [index: number]: string;
    }
    let myArray: ReadonlyStringArray = ["Alice", "Bob"];
    myArray[2] = "Mallory"; // error!
    
  • 同时设置string,number两种签名,两种类型要一致

类类型

  • 实现接口 implements
    • 类名 implements 接口
    • 接口描述了类的公共部分,而不是公共和私有两部分。 它不会帮你检查类是否具有某些私有成员。
  • 类静态部分与实例部分的区别
    下面代码会报错,因为接口无法实现检测constructor
interface ClockConstructor {
    new(hour: number, minute: number): ClockConstructor;
}
class Clock implements ClockConstructor {
}

可以定义一个函数去实现

interface Test1 {
    new(hour: number, minute: number): Test2;
}

interface Test2 {
}
// 在这里可以使用
function create(ctor: Test1, hour: number, minute: number): Test2 {
    return new ctor(hour, minute);
}

class DigitalClock implements Test2 {
}

class AnalogClock implements Test2 {
}

create(DigitalClock, 12, 17);
create(AnalogClock, 7, 32);

拓展接口

  • 可以继承,多继承
interface Shape {
    color: string;
}

interface PenStroke {
    penWidth: number;
}

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

混合类型

  • 一个函数对象带有属性
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;

接口继承类

  • 当接口继承了类,自动获取类的成员,包括方法,但是没有具体实现
  • 接口会继承类的private和protected成员
    • 此接口只能被这个类,和子类实现(implement)
class C1 {
    private key: any
}

interface I1 extends C1 {
    method(): void
}
// 接口继承了C1 但是C3没有继承C1 报错
class C3 implements I1 {
    private key: any;

    method(): void {
    }

}
// C2为C1的子类所以可以实现I1
class C2 extends C1 implements I1 {
    method(): void {
    }
}








  • 继承后constructor中第一行必须写super
    • constructor(name: string) { super(name) }

权限修饰符

  • public默认为此值
  • private 私有
    • 只能在类的内部访问
  • protected受保护
    • 只能在类和其派生类(子类)访问

readonly修饰符

  • 类属性也可以设置为只读
    • 只能在声明或构造函数被初始化
        class C {
            readonly name: string = "1"
        
            constructor() {
                this.name = "2" //此为最终值 
            }
        }
      

参数属性

  • constuctor中将实例属性,声明并赋值
    class Test {
        constructor(
            readonly name: number,
            public age: number,
            private weight: number,
            protected height: number
        ) {
            console.log(this.name, this.age, this.weight, this.height);
            //1 2 3 4
        }
    }
    new Test(1, 2, 3, 4)
    
  • public private protected readonly 都可以实现参数属性
  • 可读性差(个人感觉)

存取器

class Test {
    protected _key: number = 0

    get key() {
        console.log("get key");
        return this._key;
    }

    set key(value) {
        console.log("set key value:" + value);
        this._key = value;
    }
}
let t = new Test()
t.key = 1 + t.key
//get key
//set key value:1
  • 只带get的属性,会被自动推断为readonly
  • 不能转换为ECMAScript 3的代码

静态属性

class Test {
    static origin = {
        x: 0,
        y: 0
    }
}

抽象类 abstract

abstract class Animal {
    abstract makeSound(): void;
}
new Animal() //error
  • 抽象方法必须在派生类(子类)中实现
  • 不能创建一个抽象类实例

类当做接口用

class Point {
    x: number = 0
    y: number = 0
}

interface Point3d extends Point {
    z: number
}

let point3d: Point3d = {x: 1, y: 2, z: 3}








``

函数

函数类型

function add(x: number, y: number): number {
    return x + y
}

let myAdd = function (x: number, y: number): number {
    return x + y
}
myAdd = (x: number, y: number): number => x + y

完整的函数类型

let myAdd: (xValue: number, yValue: number) => number
// 类型参数的名字 不必和 函数参数名字相同
myAdd = function (x: number, y: number): number {
    return x + y
}
myAdd = (x: number, y: number): number => x + y

可选参数 和 默认参数

  • ?来实现可选参数
  • 可选参数必须在必须参数后面
  • 提供默认参数 , 则不能让其变为可选参数
function buildName(firstName: string, lastName: string = "默认参数") {
    return firstName + " " + lastName
}

buildName("Bob")  // Bob 默认参数
buildName("Bob", "Adams")  // Bob Adams

剩余参数...

  • 是个数组
function buildName(firstName: string, ...restOfName: string[]) {
    return firstName + " " + restOfName.join(" ")
}

let employeeName = buildName("Joseph", "Samuel", "Lucas", "MacKinzie")

重载

function test(x: number): string
function test(x: string): number
//实现方法
function test(x: any): any {
    if (typeof x === 'number') {
        return "重载"
    } else if (typeof x === 'string') {
        return 1
    }
}

console.log(test(2));
console.log(test(""));
  • 这里只有2个重载 function test(x: any): any并不是重载列表的一部分








泛型

简单使用

  • T是你传入的类型<string>
function identity<T>(arg: T): T {
    return arg
}

let output = identity<string>("myString")
console.log(output) //myString

泛型函数的使用

function identity<T>(arg: T): T {
    return arg
}

有两种方式定义

let myIdentity: <T> (arg: T) => T

用带有调用签名的对象字面量来定义泛型函数:

let myIdentity: { <T>(arg: T): T }

泛型接口

interface GenericIdentityFn {
    <T>(arg: T): T;
}

function identity<T>(arg: T): T {
    return arg;
}

let myIdentity: GenericIdentityFn = identity;

在接口上直接加上泛型,而非方法

interface GenericIdentityFn<T> {
    (arg: T): T;
}

function identity<T>(arg: T): T {
    return arg;
}

let myIdentity: GenericIdentityFn<number> = identity;

泛型类

class GenericNumber<T> {
    zeroValue: T
    add: (x: T, y: T) => T
}
  • 泛型类是实例部分,静态部分不能用泛型类

泛型约束

  • 用extends来约束泛型
  • 必须转入符合约束类型的值
interface Lengthwise {
    length: number
}

function loggingIdentity<T extends Lengthwise>(arg: T): T {
    console.log(arg.length)
    return arg
}
loggingIdentity(3);  // Error
loggingIdentity({length: 10, value: 3});

在泛型中使用类类型

  • 引用构造函数
function create<T>(c: { new(): T; }): T {
    return new c()
}

使用原型属性推断并约束构造函数与类实例的关系

class A {
    a: number
}

class B extends A {
    b: string
}

function createInstance<T extends A>(c: new() => T): T {
    return new c()
}

console.log(createInstance(B).b);








枚举

数字枚举

enum Direction {
    Up = 1,
    Down,
    Left,
    Right
}
  • 默认从0开始
  • 自增长,其余成员自动加1, Down为2,Left为3

字符串枚举

enum Direction {
    Up = "UP",
    Down = "DOWN",
    Left = "LEFT",
    Right = "RIGHT",
}
  • 没有自增长

计算枚举

enum E {
    A = Math.max(0,1),
    B = Math.abs(-2),
    Read    = 1 << 1,
    Write   = 1 << 2,
    ReadWrite  = Read | Write,
    G = "123".length,
    C, //错误 没有自增长 必须初始化
}
  • 没有自增长

枚举也是类型

enum ShapeKind {
    Circle,
    Square,
}

interface Square {
    kind: ShapeKind;
    sideLength: number;
}

const a: Square = {
    kind: ShapeKind.Square,
    sideLength: 1
}
enum E {
    Foo,
    Bar,
}

function f(x: E.Bar) {
    console.log(x);
}

f(E.Bar)

运行时的枚举

  • 运行时 , 是对象
enum E {
    X, Y, Z
}

function f(obj: { X: number }) {
    return obj.X;
}

console.log(f(E)); // 0

反向映射

  • 从枚举值->枚举名
    • 不支持字符串枚举, 只有数字枚举可以
enum Enum {
    A
}
let a = Enum.A
let nameOfA = Enum[a]
console.log(nameOfA)// "A"

const枚举

  • 为了避免在额外生成的代码上的开销和额外的非直接的对枚举成员的访问
const enum Enum {
    A = 1,
    B = A * 2
}








类型兼容

介绍

  • 结构性类型相同即可
interface Named {
    name: string
}

class Person {
    name: string
}

let p: Named
// 合法代码
p = new Person()

兼容

  • 基本规则: 要有相同属性
interface Named {
    name: string
}

let x: Named 
let y = {name: 'Alice', location: 'Seattle'}
//合法
x = y 
//不合法  不能直接赋值
x = {name: 'Alice', location: 'Seattle'}

函数

  • 简单说 大的能兼容小的 , 反之不行
let x = (a: number) => 0;
let f = (b: number, s: string) => 0;

f = x; // OK
x = f; // Error
  • 返回值
let x = () => ({name: 'Alice'});
let y = () => ({name: 'Alice', location: 'Seattle'});

x = y; // OK
y = x; // Error

类型

  • 只比较: 实例成员
  • 不比: 构造函数,静态成员
class Animal {
    feet: number;
    constructor(name: string, numFeet: number) {}
}
class Size {
    feet: number;
    constructor(numFeet: number) {}
}
let a: Animal;
let s: Size;
a = s;  // OK
s = a;  // OK








高级类型

交叉类型(Intersection Types)

  • 将多个类型叠加 T&U
 function extend<T, U>(first: T, second: U): T & U {
    let result = <T&U>{}
    for (const key in first) {
        result[key] = (<T&U>first)[key]
    }
    for (const key in second) {
        result[key] = (<T&U>second)[key]
    }
    return result
}

class Person {
    constructor(public name: string) { }
}
class ConsoleLogger  {
    log() {}
}
var jim = extend(new Person("Jim"), new ConsoleLogger());
var n = jim.name;

联合类型(Union Types)

  • 用|分隔每个类型 string|number
  • 只能访问联合联系中共有的成员
class Bird {
    fly(){}
    layEggs(){}
}
class Fish {
    swim(){}
    layEggs(){}
}

function getSmallPet(arg:string|number):Bird |Fish{
    if(typeof arg === "string"){
        //类型判断后 即可使用所有成员
        console.log(arg.charAt(0));
        return new Bird()
    } else if(typeof arg === 'number'){
        console.log(arg.toFixed(3));
        return new Fish()
    }
}

let pet = getSmallPet(1);
pet.swim() // Error

类型保护与区分类型

// 每一个成员访问都会报错
if (pet.swim) {
    pet.swim();
}
else if (pet.fly) {
    pet.fly();
}
  • 用类型断言,让代码工作
if ((<Fish>pet).swim) {
    (<Fish>pet).swim();
}else {
    (<Bird>pet).fly();
}

类型保护

  • 上面要用多次类型断言,要是一次检查类型后续分支清楚知道pet类型就好了,类型保护就为了这个目的
  • 用法 函数返回值为参数 is 类型
// 注意函数返回值
function isFish(pet: Fish | Bird): pet is Fish {
    return (<Fish>pet).swim !== undefined
}
//根据返回值的true false 判断是否为Fish类型
  • TypeScript不仅知道在 if分支里 pet是 Fish类型; 它还清楚在 else分支里,一定 不是 Fish类型,一定是 Bird类型。
if (isFish(pet)) {
    pet.swim();
}else {
    pet.fly();
}

typeof 类型保护

  • typeof也会被识别为类型保护
  • 只支持"number", "string", "boolean"或 "symbol"。 但是TypeScript并不会阻止你与其它字符串比较,但不会把那些表达式识别为类型保护。
function test(arg:any){
    if(typeof arg === 'string'){
        arg.toLocaleLowerCase()
    }
}

instanceof 类型保护

  • 右侧为构造函数
if (pet instanceof Bird) {
    pet.fly()
} else {
    pet.swim()
}

null

  • unll和undefined被认为可以复制为任何类型
  • 开启strictNUllChecks:true,声明变量时,不会自动保护null|undefined
  • 你可以用联合类型主动包含
let s = "foo";
s = null; // 错误, 'null'不能赋值给'string'
let sn: string | null = "bar";
sn = null; // 可以

sn = undefined; // error, 'undefined'不能赋值给'string | null'

可选参数和可选属性

  • 可选参数类型自动加上 |undefined
function test(a?: number) {
    console.log(a); // undefined
    return
}
test()
test(undefined) 
// 不会报错, 可选参数自动加了undefined
  • 可选属性,同样处理
class C {
    a: number;
    b?: number;
}

let c = new C();
c.a = 12;
c.a = undefined; // error, 'undefined' is not assignable to 'number'
c.b = 13;
c.b = undefined; // ok
c.b = null; // error, 'null' is not assignable to 'number | undefined'

去除null

function f(sn: string | null): string {
    if (sn == null) {
        return "default";
    } else {
        return sn;
    }
}
function f(sn: string | null): string {
    return sn || "default";
}

如果编译器不能去除nullundefined,可以使用类型断言手动去除.

语法: object!.key

function fixed(name: string | null): string {
    //嵌套函数,编译器无法知道调用时的name类型
    //用类型断言 !.
    let postfix = () => name!.charAt(0)
    name = name || "Bob"
    return postfix()
}

类型别名

  • 给类型起个新名字
type Name = string
type NameResolver = () => string
type NameOrResolver = Name | NameResolver

function getName(n: NameOrResolver): Name {
    if (typeof n === 'string') {
        return n
    } else {
        return n()
    }
}
  • 可以是泛型
type Container<T> = { value: T }
  • 可以引用自己
type Tree<T> = {
    value: T
    left: Tree<T>
    right: Tree<T>
}
  • 与 交叉类型 使用, 可以创建出奇怪的类
type LinkedList<T> = T & { next: LinkedList<T> }

interface Person {
    name: string
}

var people: LinkedList<Person>
var s = people.name
var s = people.next.name
var s = people.next.next.name
var s = people.next.next.next.name

接口 vs 类型别名

  • 重要的区别: 类型别名type不能extendsimplements.

字符串字面量类型

  • 固定的字符串值
type Easing = "ease-in" | "ease-out" | "ease-in-out";

class UIElement {
    animate(dx: number, dy: number, easing: Easing) {
        if (easing === "ease-in") {
        } else if (easing === "ease-out") {
        } else if (easing === "ease-in-out") {
        }
    }
}

let button = new UIElement();
button.animate(0, 0, "ease-in");
button.animate(0, 0, "uneasy"); // error: "uneasy" is not allowed here
  • 区分 函数重载
function createElement(tagName: "img"): HTMLImageElement;
function createElement(tagName: "input"): HTMLInputElement;

function createElement(tagName: string): Element {
}

数字字面量类型

type a = 1 | 2 | 3 | 4
let test: a = 1

可辨识联合

  • 语法规则
    • 类型都具有相同的属性— 可辨识的特征
    • 一个类型别名包含了那些类型的联合— 联合
interface Square {
    kind: "square"
    size: number
}

interface Rectangle {
    kind: "rectangle"
    width: number
    height: number
}

interface Circle {
    kind: "circle"
    radius: number
}
// 联合
type Shape = Square | Rectangle | Circle

function area(s: Shape) {
    // 相同的属性kind
    switch (s.kind) {
        case "square":return s.size
        case "rectangle":return s.height * s.width
        case "circle":return s.radius ** 2
    }
}

多态的this类型

  • 函数返回值类型为this
  • 链式调用
class BasicCalculator {
    public constructor(protected value: number = 0) {
    }

    public currentValue(): number {
        return this.value;
    }

    public add(operand: number): this {
        this.value += operand;
        return this;
    }

    public multiply(operand: number): this {
        this.value *= operand;
        return this;
    }
}

let v = new BasicCalculator(2)
    .multiply(5)
    .add(1)
    .currentValue();
  • 继承这个类,新的类使用此方法,不用做任何变化

class ScientificCalculator extends BasicCalculator {
    public constructor(value = 0) {
        super(value);
    }
    public sin() {
        this.value = Math.sin(this.value);
        return this;
    }
}

let v = new ScientificCalculator(2)
    .multiply(5)
    .sin()
    .add(1)
    .currentValue();

索引类型(Index types)

用typescript实现,通过索引类型查询索引访问

  • 索引类型查询
interface Person {
    name: string;
    age: number;
}
// 值只能为Person对象的属性
let s: keyof Person = 'name'
let a: (keyof Person)[] = ['name', 'age']
let b: Array<keyof Person> = ['name', 'age']
  • 索引访问
interface Person {
    name: string;
    age: number;
}
// 类型为Person所有属性的类型  string|number
let s: Person[keyof Person] = 'name'
s = 1
s = true //error
  • 检查动态属性

    举例: javascript中的从对象中取属性
function pluck(o, names) {
    return names.map(n => o[n]);
}

typescript:

function pluck<T, K extends keyof T>(o: T, names: K[]): T[K][] {
  return names.map(n => o[n]);
}

interface Person {
    name: string;
    age: number;
}
let person: Person = {
    name: 'Jarid',
    age: 35
};
let strings: string[] = pluck(person, ['name']); // ok, string[]
//idea中会提示错误(运行时是无错的) 
//vscode不会提示错误,对ts支持比idea更好
//idea可以这样解决
let strings = pluck<Person, keyof Person>(person, ['name'])
//但是这样就不是string[]了, 而是(number|string)[]
//因为Person的属性值age是number类型

映射类型

  • 将 类型的属性 变为 可选属性,只读属性
  • TS提供了从旧类型中 创建 新类型的一直方式 映射类型
type Readonly<T> = {
    readonly [P in keyof T]: T[P]
}
type Partial<T> = {
    [P in keyof T]?: T[P]
}
type PersonPartial = Partial<Person>
type ReadonlyPerson = Readonly<Person>

简单的映射类型

type Keys = 'option1' | 'option2'
type Flags = { [K in Keys]: boolean }
let a: Flags = {
    option1: true,
    option2: false
}
  • 将Person改下
type NullablePerson = {
    [P in keyof Person]: Person[P] | null
}
type PartialPerson = {
    [P in keyof Person]?: Person[P]
}

改为通用版本

type Nullable<T> = {
   [P in keyof T]: T[P] | null
}
type Partial<T> = {
   [P in keyof T]?: T[P]
}

注意 Readonly<T>和 Partial<T>用处不小,因此它们与 Pick和 Record一同被包含进了TypeScript的标准库里:

type Pick<T, K extends keyof T> = {
    [P in K]: T[P];
}
type Record<K extends string, T> = {
    [P in K]: T;
}

Readonly, Partial和 Pick是同态的,但 Record不是。 因为 Record并不需要输入类型来拷贝属性,所以它不属于同态:

type ThreeStringProps = Record<'prop1' | 'prop2' | 'prop3', string>

预定义的有条件类型

  • Exclude<T, U> -- 从T中剔除可以赋值给U的类型。
  • Extract<T, U> -- 提取T中可以赋值给U的类型。
  • NonNullable<T> -- 从T中剔除null和undefined。
  • ReturnType<T> -- 获取函数返回值类型。
  • InstanceType<T> -- 获取构造函数类型的实例类型。
type T00 = Exclude<"a" | "b" | "c" | "d", "a" | "c" | "f">;  // "b" | "d"
type T01 = Extract<"a" | "b" | "c" | "d", "a" | "c" | "f">;  // "a" | "c"

type T02 = Exclude<string | number | (() => void), Function>;  // string | number
type T03 = Extract<string | number | (() => void), Function>;  // () => void

type T04 = NonNullable<string | number | undefined>;  // string | number
type T05 = NonNullable<(() => string) | string[] | null | undefined>;  // (() => string) | string[]

function f1(s: string) {
    return { a: 1, b: s };
}

class C {
    x = 0;
    y = 0;
}

type T10 = ReturnType<() => string>;  // string
type T11 = ReturnType<(s: string) => void>;  // void
type T12 = ReturnType<(<T>() => T)>;  // {}
type T13 = ReturnType<(<T extends U, U extends number[]>() => T)>;  // number[]
type T14 = ReturnType<typeof f1>;  // { a: number, b: string }
type T15 = ReturnType<any>;  // any
type T16 = ReturnType<never>;  // any
type T17 = ReturnType<string>;  // Error
type T18 = ReturnType<Function>;  // Error

type T20 = InstanceType<typeof C>;  // C
type T21 = InstanceType<any>;  // any
type T22 = InstanceType<never>;  // any
type T23 = InstanceType<string>;  // Error
type T24 = InstanceType<Function>;  // Error

没有Omit<T, K>类型,因为它可以很容易的用Pick<T, Exclude<keyof T, K>>来表示。








Symbols

介绍

  • Symbol函数构建
let sym1 = Symbol();

let sym2 = Symbol("key"); // 可选的字符串key
  • Symbol是不可改变且唯一的
let sym2 = Symbol("key");
let sym3 = Symbol("key");

sym2 === sym3; // false, symbols是唯一的
  • symbol也可以被用做对象属性的键
const sym = Symbol();
// let sym = Symbol(); 用let会报错,必须用const (不清楚原因)
let obj = {
    [sym]: "value"
};

console.log(obj[sym]); // "value"
  • 类成员
let getClassNameSymbol = Symbol();

class C {
    [getClassNameSymbol](){
       return "C";
    }
}

let c = new C();
let className = c[getClassNameSymbol](); // "C"
console.log(className);








送代器和生成器

可送代性

  • 当一个对象实现了Symbol.iterator属性时,被认为可送代
  • ArrayMapSetStringInt32ArrayUint32Array等都已经实现了各自的Symbol.iterator
  • 对象上的 Symbol.iterator函数负责返回供迭代的值。
var arr = ['w', 'y', 'k', 'o', 'p'];
var eArr = arr[Symbol.iterator]();
// 您的浏览器必须支持for...of循环
// 以及let —— 将变量作用域限定在 for 循环中
for (let letter of eArr) {
  console.log(letter); //'w', 'y', 'k', 'o', 'p'
}
  • 另一种方法
var arr = ['w', 'y', 'k', 'o', 'p'];
var eArr = arr[Symbol.iterator]();
for (let next = eArr.next(); !next.done; next=eArr.next()) {
    console.log(next.value);
}

for..of语句

  • 遍历可迭代的对象,调用对象上的Symbol.iterator方法
  • for..of则迭代对象的键对应的值
let someArray = [1, "string", false];
for (let entry of someArray) {
    console.log(entry); // 1, "string", false
}

for..in语句

  • for..in迭代的是对象的 键 的列表
let list = [4, 5, 6];
for (let i in list) {
    console.log(i); // "0", "1", "2",
}








模块

导出声明

  • 任何声明(比如变量,函数,类,类型别名或接口)都能够通过添加export关键字来导出
export interface StringValidator {
    isAcceptable(s: string): boolean;
}
export const numberRegexp = /^[0-9]+$/;

export class ZipCodeValidator implements StringValidator {
    isAcceptable(s: string) {
        return s.length === 5 && numberRegexp.test(s);
    }
}

导出语句

  • 可以对导出的部分重新命名
class ZipCodeValidator {
}
export { ZipCodeValidator };
export { ZipCodeValidator as mainValidator };

重新导出

//从module文件中拿出 重新导出 重命名为 T在导出
export {重新导出 as T} from "./module";
  • 将所有文件内export的内容,联合导出
export * from './module'
//另一个文件 导入
import * as name from "./test2"

导入

  • 导入某个导出内容
import { ZipCodeValidator } from "./ZipCodeValidator"
  • 对导入内容重命名
import { ZipCodeValidator as ZCV } from "./ZipCodeValidator"
  • 将整个模块导入到一个变量
import * as validator from "./ZipCodeValidator"
  • 有副作用的导入模块
import "./my-module.js"

默认导出

每个模块都可以有一个且只能有一个default导出

let x = 1
export default x
//在其他文件中导入 x 自定名字
import 自定义名字 from './module'
  • 函数和声明可以直接被标记为默认导出
export default class a {}
//函数
export default function () {}
  • 导出也可以是一个值
export default '123'

动态模块加载

declare function require(moduleName: string): any;

import {Zip} from "./module"

setTimeout(() => {
    Zip()
    let test2= require("./test2");
    console.log(test2.f1());
}, 1000)

使用其他JavaScript库

想描述非TypeScript编写的类库的类型,我们需要声明类库所暴露出的API。

我们叫它声明因为它不是“外部程序”的具体实现。 它们通常是在 .d.ts文件里定义的。 如果你熟悉C/C++,你可以把它们当做 .h文件。 让我们看一些例子。

外部模块

在Node.js里大部分工作是通过加载一个或多个模块实现的。 我们可以使用顶级的 export声明来为每个模块都定义一个.d.ts文件,但最好还是写在一个大的.d.ts文件里。 我们使用与构造一个外部命名空间相似的方法,但是这里使用 module关键字并且把名字用引号括起来,方便之后import。 例如:

declare module "url" {
    export interface Url {
        protocol?: string;
        hostname?: string;
        pathname?: string;
    }

    export function parse(urlStr: string, parseQueryString?, slashesDenoteHost?): Url;
}

declare module "path" {
    export function normalize(p: string): string;
    export function join(...paths: any[]): string;
    export let sep: string;
}

现在我们可以/// <reference> node.d.ts并且使用import url = require("url");或import * as URL from "url"加载模块。

/// <reference path="node.d.ts"/>
import * as URL from "url";
let myUrl = URL.parse("http://www.typescriptlang.org");








声明合并

合并接口

合并的机制是把双方的成员放到一个同名的接口里。

interface Box {
    height: number;
    width: number;
}

interface Box {
    scale: number;
}

let box: Box = {height: 5, width: 6, scale: 10};

接口的非函数的成员应该是唯一的。如果它们不是唯一的,那么它们必须是相同的类型。如果两个接口中同时声明了同名的非函数成员且它们的类型不同,则编译器会报错。

对于函数成员,每个同名函数声明都会被当成这个函数的一个重载。 同时需要注意,当接口 A与后来的接口 A合并时,后面的接口具有更高的优先级。

如下列所示:

interface Cloner {
    clone(animal: Animal): Animal;
}

interface Cloner {
    clone(animal: Sheep): Sheep;
}

interface Cloner {
    clone(animal: Dog): Dog;
    clone(animal: Cat): Cat;
}

这相当于

interface Cloner {
    clone(animal: Dog): Dog;
    clone(animal: Cat): Cat;
    clone(animal: Sheep): Sheep;
    clone(animal: Animal): Animal;
}

合并命名空间

对于导出的成员进行合并

namespace Animals {
    export class Zebra { }
}

namespace Animals {
    export interface Legged { numberOfLegs: number; }
    export class Dog { }
}

等同于

namespace Animals {
    export interface Legged { numberOfLegs: number; }

    export class Zebra { }
    export class Dog { }
}

非导出成员,仅在其原有的(合并前的)命名空间内可见。

这就是说合并之后,从其它命名空间合并进来的成员无法访问非导出成员。

namespace Animal {
    let haveMuscles = true;

    export function animalsHaveMuscles() {
        return haveMuscles;
    }
}

namespace Animal {
    export function doAnimalsHaveMuscles() {
        return haveMuscles;  // Error,访问不到haveMuscles
    }
}

由于装饰器还有可能会进行改动,所以暂时观望。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容