0

0

如何在 JavaScript 中实现对可选 fetch 请求的单次懒加载调用

霞舞

霞舞

发布时间:2026-02-24 18:55:28

|

154人浏览过

|

来源于php中文网

原创

如何在 JavaScript 中实现对可选 fetch 请求的单次懒加载调用

本文介绍一种高效、线程安全的方式,通过 promise 缓存与空值合并赋值(??=)操作符,确保同一可选网络请求(如 fetch('c'))在整个生命周期中最多执行一次,避免重复请求和资源浪费。

本文介绍一种高效、线程安全的方式,通过 promise 缓存与空值合并赋值(??=)操作符,确保同一可选网络请求(如 fetch('c'))在整个生命周期中最多执行一次,避免重复请求和资源浪费。

在前端开发中,常遇到一类典型场景:多个逻辑分支可能条件性触发同一个耗时网络请求(例如配置获取、权限校验或共享数据接口),但该请求实际只需执行一次——后续调用应直接复用首次结果。若不加控制,极易导致重复 fetch,不仅增加服务器压力、延长首屏时间,还可能因响应顺序不确定引发状态不一致问题。

上述问题的核心诉求是:对可选请求 'c' 实现懒加载(lazy)、单例化(singleton)、Promise 级缓存(promise-level caching)。关键在于:

梯子AI
梯子AI

百度推出的AI智能搜索

下载
  • 首次调用 c() 时发起真实 fetch 并返回其 Promise;
  • 后续调用 c() 应直接返回同一个 Promise 实例(而非新建 Promise),从而天然共享 resolve 结果与错误状态;
  • 无需手动管理 loading 状态或锁机制,利用 JavaScript 的 Promise 特性即可保证线程安全。

✅ 推荐实现:利用 ??= 运算符缓存 Promise

let cPromise;

const c = () => (cPromise ??= fetch('c').then(r => r.json()));

// 分支 A:根据 a 的响应决定是否调用 c
fetch('a')
  .then(r => r.json())
  .then(data => {
    if (data.value === '1') {
      c().then(result => console.log('through a', result));
    }
  });

// 分支 B:根据 b 的响应决定是否调用 c
fetch('b')
  .then(r => r.json())
  .then(data => {
    if (data.value === '1') {
      c().then(result => console.log('through b', result));
    }
  });

? 关键原理说明:

  • cPromise ??= ... 是 Nullish Coalescing Assignment(空值合并赋值),仅当 cPromise 为 null 或 undefined 时才执行右侧表达式并赋值;
  • fetch('c').then(...) 返回一个 Promise 实例,该实例被一次性赋给 cPromise,后续所有 c() 调用都返回这个已存在的 Promise;
  • 即使 fetch('c') 尚未完成,c() 仍返回同一个 pending Promise,因此 .then() 注册的回调会自动在 resolve 后按注册顺序执行——完全符合预期行为。

⚠️ 注意事项与最佳实践

  • 不要在 c() 内部 new Promise(...) 包裹 fetch:这会创建新 Promise,破坏缓存语义,导致重复请求;
  • 避免使用 let cPromise = null + if (!cPromise) cPromise = ...:在并发调用下存在竞态风险(两个 c() 同时判断 !cPromise 为真),而 ??= 是原子操作,无此问题;
  • 错误处理需统一捕获:由于所有调用共享同一 Promise,建议在 c() 外层或首次调用时添加 .catch(),或在 c() 中返回 fetch(...).then(...).catch(...) 显式处理错误;
  • 适用范围:本方案适用于「结果不变」或「幂等」的请求(如静态配置、只读元数据)。若请求结果随时间变化(如实时计数器),则需额外引入 TTL 或手动失效机制;
  • ES2021+ 兼容性:??= 需现代浏览器或构建工具(如 Babel)支持;若需兼容旧环境,可用 cPromise || (cPromise = ...) 替代(注意此处用 || 而非 ??,因 Promise 永不为 falsy)。

✅ 总结

通过一个轻量级的闭包变量 cPromise 与 ??= 运算符,我们以最小认知成本实现了「条件触发、最多一次、自动复用」的 fetch 策略。它不依赖第三方库,不增加运行时开销,且完全符合 Promise 规范——是解决此类问题的专业、简洁、可靠的首选方案。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

246

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

866

2024.03.01

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1558

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

239

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

127

2025.10.17

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

829

2023.08.22

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1630

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

485

2025.10.17

Golang 生态工具与框架:扩展开发能力
Golang 生态工具与框架:扩展开发能力

《Golang 生态工具与框架》系统梳理 Go 语言在实际工程中的主流工具链与框架选型思路,涵盖 Web 框架、RPC 通信、依赖管理、测试工具、代码生成与项目结构设计等内容。通过真实项目场景解析不同工具的适用边界与组合方式,帮助开发者构建高效、可维护的 Go 工程体系,并提升团队协作与交付效率。

1

2026.02.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 5.4万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 3.1万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号