TS项目中如何合理的为接口定义参数类型

关于ts项目中如何合理的为后端接口定义参数类型

复制代码
在传统业务的ts项目中,通常会调用很多后端的api,而几乎每个api的入参和出参都不一样,通常的做法是人为的为每个api定义对呀的参数类型,不仅工作量大,而且重复率高

市面上一些常见的方案是这样的:

  1. 为api入参或者出参定义为any类型
  2. 简单粗暴的为每个api定义为ts类型
  3. 使用工具去提前调用接口,然后根据接口返回参数生成对呀的ts类型
  4. 后端使用node+ts开发,或者增加node+ts的中间层,这在微服务架构中叫BFF层,其作用就是聚合基础api字段,返回业务数据,因为在后端使用ts提供接口的时候,必然需要主动去声明api的入参和出参类型,因此就可以通过代码共享的方式和后端共用ts数据类型

其中1、2点不过多解释;

第3点应该有对应实现工具,但是缺点很明显

  • 一是接口返回的类型可能不全,而且调用接口传递模拟参数比较麻烦
  • 二是类型的重复率还是很高,框架在做类型检查的时候还是会比较消耗性能
  • 三是工具生成的类型代码在源码中不具备任何意义

第4点是比较好的做法,但是也有一些缺点

  • 一是通常在规模比较大的项目中实现效果更好,因为比较规范的微服务架构以及DDD架构思维事很难落地的,其中包括领域的合理划分,以及服务职责权限等,更别提BFF在项目中的实现
  • 二是即使在后端架构中存在BFF层,也不能保证BFF层会涉及到ts代码

将数据库字段类型定义在ts类型中

复制代码
一些思考:前端项目中的三架马车css、html、js,前两者暂不谈,js开发的项目和其它所有开发语言开发的项目一样,都是由变量、常量、关键字这三部分组成,常量和关键字都是不可变的,剩下的变量都具有继承或者类继承的特性,总能在变量上找到它的原型或者构造它的函数,像js中所有对象最终的原型都指向null一样,似乎在业务代码中也应该使用这样理想的开发模式;

想要定义合理的出入参类型的前提是先合理划分后端微服务子域类型,只有在保证后端逻辑清晰的情况下才能更好的在前端定义对应的参数类型,以普通商城项目为例,后端可能划分有商品、物料、用户、订单等子域,对应的前端也应该划分相应的数据类型,下面为简单例子

typescript 复制代码
/* user.ts */
export type Id = number
export type Name = string
export type Age = number
export interface User {
  id: Id //用户id
  name: Name
  age: Age
}
typescript 复制代码
/* product.ts */
export type Id = string
export type ImgUrl = string
export type Price = number
export type Title = string
export interface Product {
  id: Id//商品id
  imgUrl: ImgUrl//商品图片
  price: Price//价格
  title: Title//商品标题
}
typescript 复制代码
/* order.ts */
import { Id as UserId } from "./user.ts"
import { Id as ProductId } from "./product.ts"

export type Id = number
export type CreateTime = string
export type Payment = number
export interface Order {
  id: Id//订单id
  createTime: CreateTime//创建时间
  payment: Payment//付款金额
  userId: UserId//用户id
  productId: ProductId//商品id
}

api /getOrderById 通过orderId来获取用来获取订单数据

ts 复制代码
import { Id, Order } from "./order.ts"
//入参类型
type E = Id
//出参类型
type P = Order

如果返回的类型是订单信息和商品信息的集合,则只用将Product和Order类型合并即可

ts 复制代码
//如果返回的类型为下面这种
const data = {
  id: 1,//订单id
  createTime: "2022-10-22",//创建时间
  payment: 33,//付款金额
  userId: 111,//用户id
  productId: "qwertyuiop",//商品id
  imgUrl: "",//商品图片
  title: ""//商品标题
}

//参数类型也可以是先排除product id 以后和order类型交叉合并
type T = Omit<Product, 'id' | 'price'> & Order
相关推荐
Highcharts.js2 小时前
Highcharts 云端渲染的真相:交互式图表与服务器端生成的边界
前端·信息可视化·服务器渲染·highcharts·图表渲染
zhuyan1083 小时前
Linux 系统磁盘爆满导致无法启动修复指南
前端·chrome
编程牛马姐4 小时前
独立站SEO流量增长:提高Google排名的优化方法
前端·javascript·网络
NotFound4864 小时前
实战指南如何实现Java Web 拦截机制:Filter 与 Interceptor 深度分享
java·开发语言·前端
Dontla4 小时前
高基数(High Cardinality)问题介绍(Prometheus、高基数字段、低基数字段)
前端·数据库·prometheus
whuhewei6 小时前
为什么客户端不存在跨域问题
前端·安全
妮妮喔妮7 小时前
supabase的webhook报错
开发语言·前端·javascript
qq_12084093717 小时前
Three.js 大场景分块加载实战:从全量渲染到可视集调度
开发语言·javascript·数码相机
yivifu7 小时前
手搓HTML双行夹批效果
前端·html·html双行夹注
奔跑的卡卡8 小时前
Web开发与AI融合-第一篇:Web开发与AI融合的时代序幕
前端·人工智能