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
相关推荐
xiao-xiang1 分钟前
jenkins-通过api获取所有job及最新build信息
前端·servlet·jenkins
C语言魔术师18 分钟前
【小游戏篇】三子棋游戏
前端·算法·游戏
小周不摆烂24 分钟前
探索JavaScript前端开发:开启交互之门的神奇钥匙(二)
javascript
匹马夕阳1 小时前
Vue 3中导航守卫(Navigation Guard)结合Axios实现token认证机制
前端·javascript·vue.js
你熬夜了吗?2 小时前
日历热力图,月度数据可视化图表(日活跃图、格子图)vue组件
前端·vue.js·信息可视化
我想学LINUX2 小时前
【2024年华为OD机试】 (A卷,100分)- 微服务的集成测试(JavaScript&Java & Python&C/C++)
java·c语言·javascript·python·华为od·微服务·集成测试
screct_demo3 小时前
詳細講一下在RN(ReactNative)中,6個比較常用的組件以及詳細的用法
javascript·react native·react.js
桂月二二8 小时前
探索前端开发中的 Web Vitals —— 提升用户体验的关键技术
前端·ux
CodeClimb9 小时前
【华为OD-E卷 - 第k个排列 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
hunter2062069 小时前
ubuntu向一个pc主机通过web发送数据,pc端通过工具直接查看收到的数据
linux·前端·ubuntu