ts中如何描述一个复杂函数的类型

函数重载

函数重载可以描述复杂的函数类型

ts 复制代码
function fn(num: number): number
function fn(str: string): string

function fn(arg: number | string) {
  if (typeof arg === 'number') return 0
  return '1'
}

不过这种方法适用范围有限 不能导出一个类型用于定义其他函数 而且需要把重载的类型在参数里额外写一次

函数的调用类型

也可以直接描述其调用类型

ts 复制代码
type FN = {
  (num: number): number
  (str: string): string
}

但这个类型只能用于描述函数 不能用于定义函数 会让ts对类型产生误判

ts 复制代码
const fn: FN = (arg) => {
  if (typeof arg === 'number') return 0
  return '1'
}

如果将回调函数声明为这个类型 写代码的时候会完全失去类型提示

TypeMap构造函数参数

ts 复制代码
type ArgMap = {
  number: { num: number }
  string: { str: string }
}

type FN = (
  config: {
    [K in keyof ArgMap]: ArgMap[K] & {
      type: K
    }
  }[keyof ArgMap],
) => void

const fn: FN = (config) => {
  switch (config.type) {
    case 'string':
      console.log(config.str)
      break
    case 'number':
      console.log(config.num)
  }
}

可以利用类型索引自由构造不同类型的参数 这里构造为对象并用type区分并不强制 甚至可以构造args数组

类型非常完备 但无法约束返回值

TypeMap+泛型

typescript 复制代码
type FN = <K extends keyof FnTypeMap>(type: K, value: FnTypeMap[K]['in']) => FnTypeMap[K]['out']

const fn: FN = (type, value) => {
  switch (type) {
    case 'number': {
      value // FnTypeMap[K]["in"] 并没有收窄类型
      break
    }
    case 'string': {
      break
    }
  }
}

类型依旧不完备

结论

复杂函数类型在ts中没有一个完美的解决方案 定义函数时重载即可 需要以一个类型约束函数的时候(例如描述回调函数或者使用zustand时)使用TypeMap构造参数对象 可以用onComplete等替代返回值

相关推荐
hunterandroid9 小时前
[Android 从零到一] 权限管理:运行时权限与最佳实践
前端
kyrie289 小时前
Redux 完整基础操作(原生 Redux,不结合 React-Redux)
前端
因_崔斯汀9 小时前
Vue 模板编译:HTML 是怎么变成 JS 的?
前端·vue.js
UXbot9 小时前
帮助企业低门槛开展AI应用开发的平台推荐
前端·低代码·ui·交互·产品经理·原型模式·web app
橘子星10 小时前
基于 Vite 的多模态生图前端工程实践
前端·javascript·人工智能
想要成为糕糕手10 小时前
从零到一:CSS 3D 旋转立方体完全指南
前端·css·canvas
疯狂的魔鬼10 小时前
多角色督办任务详情页:从权限矩阵到组件拆分的完整实现
前端·vue.js·架构
恋猫de小郭10 小时前
Android 17 正式版发布,全新 AI 和各种破坏性更新
android·前端·flutter
半个落月10 小时前
从零搭建 AI 生图前端|Vite 工程化 + 通义千问 API 实战,附 API Key 安全方案
前端·人工智能
退休倒计时10 小时前
【每日一题】LeetCode 146. LRU 缓存 TypeScript
算法·leetcode·缓存·typescript