高可维护性TS编码技巧

  • Post author:
  • Post category:其他




高可维护性TS编码技巧


1.显示的定义要使用的类型 减少借助类型推断的类型

import { useState } from 'react'

interface IUser {
  name: string;
}

function Component(user: IUser) {
  // 即使可以类型推到任然显示的去定义类型
  const [userInfo, setUserInfo] = useState<IUser>(user)
}


2. 合理使用enum和type

/ 通常来说,enum和type的使用场景还是比较接近的

// 在字段值是无语义的时候建议用enum
enum PersonType {
    plocieman = 0,
    firman = 1
}

// 如果字段值本身就是有语义的东西的时候建议用type
type KeyName = 'address' | 'name' | 'age'

// 通过const减少enum编译的内容, 在想把打包后代码体积压缩到极致的时候会有一定帮助
const enum PersonType2 {
    plocieman = 0,
    firman = 1
}

在这里插入图片描述

在这里插入图片描述


3. 利用keyof快速生成关键字

interface A {
  age: number;
  name: string;
}

type AKeys = keyof A; // age | name


4. 使用ts提供的原生类型辅助工具

// Partial 将类型所有属性转化为可选的
interface A {
    age: number;
    name: string;
}

/**
** {age?:number;name?:string}
**/
type PartialA = Partial<A>
// Required 将类型所有属性转化为必选的
interface A {
    age?: number;
    name?: string;
}

/**
** {age:number;name:string}
**/
type RequiredA = Required<A>

// Readonly 将类型所有属性转化为只读的
interface A {
    age: number;
    name: string;
}

/**
** {readonly age:number;readonly name: string}
**/
type ReadOnlyA = ReadOnly<A>
// Pick 提取出想要的属性组成一个新的类型
interface A {
    age: number;
    name: string;
    address: string;
}

/**
** {age:number;address: string}
**/
type PickA = Pick<A, 'age'|'address'>
// Omit 剔除掉不想要的属性组成一个新的类型
interface A {
    age: number;
    name: string;
    address: string;
}

/**
** {age: number;address: string;}
**/
type OmitA = Omit<A, 'name'>

// Extract 提取两个类型的交集

type A = 'name' | 'age'
type B = 'age' | 'address'

// 'age'
type C = Extract<A, B>
// NonNullable 剔除掉null和undefined类型

type A = string | number | null | undefined
// string | number
type B = NonNullable<A>


5. 利用is帮助类型推断

interface A {
  type: string;
  data: {
    name: string;
  }
}
interface B {
  type: string;
  data: boolean;
}
function isA(params: any): boolean {
  return params.type === 'A';
}
// 这里能够帮助推断出类型
function isA2(params: any): params is A {
  return params.type === 'A';
}
function test(params: A | B) {
  if (isA(params)) {
    (params as A).data
  }
  if ((isA2(params))) {
    params.data
  }
}


6. 泛型类型限制

// 这里就会限制了类型A的data属性必须是有length属性,且length类型为number
interface A<T extends {length: number;}> {
  data:T;
}


7. Infer 辅助类型推断

// ts 官方是有提供相关的几个工具类型,也有用到 infer

/**
 * Obtain the parameters of a constructor function type in a tuple
 */
type ConstructorParameters<T extends new (...args: any) => any> = T extends new (...args: infer P) => any ? P : never;

/**
 * Obtain the return type of a function type
 */
type ReturnType<T extends (...args: any) => any> = T extends (...args: any) => infer R ? R : any;


8. 函数重载

// 常见的场景就是我们根据不同传参类型返回不同的值
interface Create {
  (val: number): number
  (val: string): string
  (val: Date): Date
}

// 这种写法无法感知传参和返回体之间的联系
// function create(params: number | string | Date): number | string | Date {}

const create: Create = (val) => {
    if (typeof val === 'number') return 0
    if (typeof val === 'string') return '0'
    return new Date(0)
}

一种比较常用的场景 eventBus

enum EventName {
  sayName,
  run,
}
interface EventBusOn {
  (event: EventName.sayName, cb: (params: {name: string}) => void): void
  (event: EventName.run, cb: (params: {duration: number}) => void): void
}

interface EventBusEmit {
  (event: EventName.sayName, params: {name: string}): void
  (event: EventName.run, params: {duration: number}): void
}

interface EventBus {
  on: EventBusOn;
  emit: EventBusEmit;
}

const eventBus: EventBus = new Vue();
eventBus.on(EventName.sayName, ({name}) => {
  console.log(name);
});
eventBus.emit(EventName.sayName, {name: 'sd'});

这样还有个问题,就是我们每次增加一个事件类型,都需要去修改 EventBusOn 和 EventBusEmit 两个接口,但是他们的修改模式基本是固定的,结合前面的两个技巧,我们也可以优化这个接口。

enum EventName {
  sayName,
  run,
  hi,
}
interface EventParams {
  [EventName.sayName]: {
    name: string;
  };
  [EventName.run]: {
    duration: number;
  };
  [EventName.hi]: {name:string}
}
interface EventBusOn {
  <T extends EventName>(event: T, cb: (params: EventParams[T]) => void): void;
}

interface EventBusEmit {
  <T extends EventName>(event: T, params: EventParams[T]): void;
}
interface EventBus {
  on: EventBusOn;
  emit: EventBusEmit;
}

const eventBus: EventBus = new Vue();
eventBus.on(EventName.sayName, ({name}) => {
  console.log(name);
})
eventBus.emit(EventName.sayName, {name: 'name'})

还遗留了一个小问题,没有解决不传参的eventBus场景,感兴趣的可以自己尝试解决一下


不传参EventBus解法一

export enum EventName {
  doSomething= 'doSomething',
  add = 'add',
  delete = 'delete'
}

interface EventParams {
  [EventName.doSomething]: DoSomething
  [EventName.add]: void
  [EventName.delete]: {name: string}
}

interface EventBusOn {
  <EventType extends EventName>(
    event: EventType,
    cb: EventParams[EventType] extends void
      ? () => void :
      (params: EventParams[EventType]) => void
  ): void
}

interface EventBusEmit {
  <EventType extends EventName>(
    ...args: EventParams[EventType] extends void
      ? [EventType] :
      [EventType, EventParams[EventType]]
  ): void
}


不传参EventBus解法二

interface EventParams {
  [EventName.doSomething]: DoSomething
  [EventName.add]: void
  [EventName.delete]: {name: string}
}

type NoParamsEvent = EventName.add |  EventName.delete

interface EventBusOn {
  <T extends Exclude<EventName, NoParamsEvent>>(event: T, cb: (params: EventParams[T]) => void): void
  <T extends NoParamsEvent>(event: T, cb: () => void): void
}

interface EventBusEmit {
  <T extends Exclude<EventName, NoParamsEvent>>(event: T, params: EventParams[T]): void
  <T extends NoParamsEvent>(event: T): void
}



版权声明:本文为CallMeBY原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。