重复请求优化

设想一种场景,有两个组件,这两个组件在初始化阶段,都需要调用同一个 API 接口去获取数据。为了防止请求冗余,可以把两个组件的请求都挪到父组件中,由父组件统一调用一次请求,然后再将响应的数据结果传给两个子组件。这种方法应该是最常见的,不过它也有一个局限性条件:两个组件必须有一个共同的祖先组件,如果这两个组件是同级的兄弟组件倒也还好,如果非同级,那么数据的传参就会有些麻烦了。那么还有其他办法吗?当然是有的。

我们可以换个思路,每个组件还是保持原有的业务逻辑不变,从请求接口处做文章。既然是同一个接口调用了两次,而且还是返回了相同的请求结果,那么不妨在第一次时调用成功时,就把请求结果缓存起来,等到第二次再调用时,直接返回缓存的数据。按照这个思路可以写出第一版的代码(这里用了 TS 方便查看参数的类型):

ts 复制代码
/**
 * 缓存请求的响应结果
 * 把发起请求的 Promise 对象挂载在原型对象上
 * @param request 请求函数
 */
function cacheRequest<T>(request: (...args: any[]) => Promise<T>) {
  const cache = Symbol("cache")
  return function (...args: any[]): Promise<T> {
    if (!request.prototype[cache]) {
      request.prototype[cache] = request(...args)
    }
    return request.prototype[cache]
  }
}
  • 首先 cacheRequest 函数,需要接收一个参数 requestrequest 是一个返回结果为 Promise 对象的函数。cacheRequest 执行完后返回一个新的匿名函数。
  • 然后,在匿名函数的内部,先判断 request 的原型对象上是否有 cache(这里的 cache 使用了 Symbol 类型,确保键名唯一)。也即,是否有缓存过的请求结果,如果没有,说明是第一次调用,则将 request 的执行结果存到缓存里。如果有缓存,则直接返回缓存。
  • 可以看到,缓存也是一个 Promise 类型。在同时调用多次请求时,只要在第一次调用执行后,已经把 Promise 存到缓存里了,后续的请求返回的也是缓存里的 Promise,从而保证多个请求都指向同一个 Promise ,也即只会调用一次 API 接口。
  • 这里需要注意一点,由于需要往 request 的原型对象上挂载缓存,所以 request 不能是箭头函数。因为箭头函数没有 this,也就意味着没有原型对象。

小测一下:

js 复制代码
function cacheRequest(request) {
  const cache = Symbol("cache")
  return function (...args) {
    if (!request.prototype[cache]) {
      request.prototype[cache] = request(...args)
    }
    return request.prototype[cache]
  }
}

const request = function () {
  return new Promise(resolve => {
    console.log("fetch request")
    setTimeout(resolve, 2000)
  })
}

const newRequest = cacheRequest(request)

newRequest()
newRequest()
newRequest()

可以看到虽然 newRequest 调用了三次,但是 fetch request 只打印了一次,也就是说 request 只调用了一次,符合预期!但是,最后一次 newRequest 的调用,是在 3 秒后调用的,也是走的缓存,没有重新执行。仔细思考一下,后续无论什么时候调用 newRequest 都会使用缓存里的数据,不会重新调用请求了,这显然是不合理的。我们还需要加个缓存的过期时间,超过这个时间,就重新发起新的请求。第二版如下:

ts 复制代码
/**
 * 缓存请求的响应结果
 * 把发起请求的 Promise 对象挂载在原型对象上
 * 保证在 cacheTime 时间间隔内的多次请求,只会调用一次
 * @param request 请求函数
 * @param cacheTime 最大缓存时间(单位毫秒)
 */
export function cacheRequest<T>(request: (...args: any[]) => Promise<T>, cacheTime = 1000) {
  const cache = Symbol("cache")
  const lastTime = Symbol("lastTime")
  return function (...args: any[]): Promise<T> {
    if (!request.prototype[cache] || Date.now() - request.prototype[lastTime] >= cacheTime) {
      request.prototype[cache] = request(...args)
      request.prototype[lastTime] = Date.now()
    }
    return request.prototype[cache]
  }
}
  • 首先,cacheRequest 新增一个入参 cacheTime,用于设置过期时间,默认为 1 秒。
  • 其次,在原型对象上新增了一个 lastTime 属性,用来记录最后一次调用的时间。
  • 当缓存为空,或者当前时间距离上一次调用时间超过缓存过期时间时,更新 cachelastTime

再来小测一下:

js 复制代码
function cacheRequest(request) {
  const cache = Symbol("cache")
  const lastTime = Symbol("lastTime")
  return function (...args) {
    if (!request.prototype[cache] || Date.now() - request.prototype[lastTime] >= cacheTime) {
      request.prototype[cache] = request(...args)
      request.prototype[lastTime] = Date.now()
    }
    return request.prototype[cache]
  }
}

const request = function () {
  return new Promise(resolve => {
    console.log("fetch request")
    setTimeout(resolve, 2000)
  })
}

const newRequest = cacheRequest(request)

newRequest()
newRequest()
setTimeout(newRequest, 3000)

这一次,fetch request 打印了两次,符合预期,完美!

相关推荐
2501_920931701 小时前
React Native鸿蒙跨平台采用ScrollView的horizontal属性实现横向滚动实现特色游戏轮播和分类导航
javascript·react native·react.js·游戏·ecmascript·harmonyos
东东5163 小时前
智能社区管理系统的设计与实现ssm+vue
前端·javascript·vue.js·毕业设计·毕设
catino3 小时前
图片、文件的预览
前端·javascript
2501_920931704 小时前
React Native鸿蒙跨平台实现推箱子游戏,完成玩家移动与箱子推动,当所有箱子都被推到目标位置时,玩家获胜
javascript·react native·react.js·游戏·ecmascript·harmonyos
AI老李5 小时前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
方也_arkling5 小时前
Element Plus主题色定制
javascript·sass
2601_949809596 小时前
flutter_for_openharmony家庭相册app实战+我的Tab实现
java·javascript·flutter
Up九五小庞6 小时前
开源埋点分析平台 ClkLog 本地部署 + Web JS 埋点测试实战--九五小庞
前端·javascript·开源
摘星编程6 小时前
React Native + OpenHarmony:UniversalLink通用链接
javascript·react native·react.js
qq_177767376 小时前
React Native鸿蒙跨平台数据使用监控应用技术,通过setInterval每5秒更新一次数据使用情况和套餐使用情况,模拟了真实应用中的数据监控场景
开发语言·前端·javascript·react native·react.js·ecmascript·harmonyos