0

0

JavaScript异步数据缓存:实现单次查询与数据复用

心靈之曲

心靈之曲

发布时间:2025-10-01 10:48:08

|

947人浏览过

|

来源于php中文网

原创

JavaScript异步数据缓存:实现单次查询与数据复用

本文探讨了如何在JavaScript异步编程中避免重复的数据库查询,实现数据的高效复用。通过介绍空值合并赋值运算符(??=)结合Promise的延迟初始化机制,我们将展示如何仅执行一次异步数据获取操作,并将返回的Promise对象缓存起来,供后续函数重复使用,从而显著提升应用性能并减少数据库负载。

异步数据获取中的重复查询问题

在现代web应用开发中,与数据库或其他外部api进行异步交互是常见的操作。为了优化性能和减少服务器负载,我们通常希望对相同的数据只进行一次查询,然后将结果缓存起来供后续函数使用。然而,在不恰当的实现下,我们可能会遇到重复查询的问题。

考虑以下JavaScript代码片段,它尝试通过一个全局变量来缓存异步获取的数据:

async function GetCoverage(scroll_path) {
  // 模拟一个异步DB或API调用
  console.log(`Executing GetCoverage for: ${scroll_path}`);
  return new Promise(resolve => {
    setTimeout(() => {
      resolve([{ id: 1, path: scroll_path, data: Math.random() }]);
    }, 500);
  });
}

let dataGlobal; // 用于存储数据或Promise

const getData = async () => {
  const response = await GetCoverage("all"); // 每次调用都会触发GetCoverage
  dataGlobal = response;
  return response;
};

(async () => {
  console.log("Initial call to getData...");
  await getData();
  console.log("dataGlobal after first call:", dataGlobal);

  console.log("\nSecond call to getData...");
  await getData(); // 这会再次调用GetCoverage
  console.log("dataGlobal after second call:", dataGlobal);
})();

在上述代码中,getData 函数每次被调用时,都会执行 await GetCoverage("all"),这意味着无论 dataGlobal 是否已经有值,GetCoverage 函数(模拟数据库查询)都会被重新触发。这导致了不必要的数据库查询,浪费了资源并增加了响应时间。

解决方案:Promise的延迟初始化与缓存

为了解决重复查询的问题,我们需要一种机制来:

  1. 延迟初始化:仅在数据首次被请求时才执行异步查询。
  2. 缓存Promise:将异步查询返回的 Promise 对象缓存起来,而不是等待它解析后的数据。这样,后续的请求可以直接返回这个已缓存的 Promise,无论它是处于待定(pending)、已解决(resolved)还是已拒绝(rejected)状态。

JavaScript中的空值合并赋值运算符 (??=) 提供了一种简洁高效的方式来实现这一目标。

立即学习Java免费学习笔记(深入)”;

空值合并赋值运算符 (??=) 简介

??= 运算符是一个逻辑赋值运算符。它的语法是 a ??= b,等同于 a = a ?? b。这意味着如果 a 是 null 或 undefined(即“空值”),那么 a 将被赋值为 b 的值;否则,a 的值保持不变。

文希AI写作
文希AI写作

AI论文写作平台

下载

利用 ??= 运算符的这一特性,我们可以非常优雅地实现Promise的延迟初始化和缓存。

实现单次查询与数据复用

我们将 dataGlobal 变量用于存储 GetCoverage 返回的 Promise。getData 函数的职责将变为:如果 dataGlobal 尚未被赋值(即为 undefined 或 null),则调用 GetCoverage 并将其返回的 Promise 赋值给 dataGlobal;否则,直接返回 dataGlobal 中已存在的 Promise。

// 假设 GetCoverage 函数已定义,与上面示例相同
async function GetCoverage(scroll_path) {
  console.log(`Executing GetCoverage for: ${scroll_path}`);
  return new Promise(resolve => {
    setTimeout(() => {
      resolve([{ id: 1, path: scroll_path, data: Math.random() }]);
    }, 500);
  });
}

let dataGlobalPromise; // 用于存储GetCoverage返回的Promise

const getData = () => (dataGlobalPromise ??= GetCoverage("all")); // 核心优化

// 演示如何使用
(async () => {
  console.log("Making parallel calls to getData...");

  // 第一次调用 getData() 会触发 GetCoverage("all")
  // 并将返回的 Promise 赋值给 dataGlobalPromise
  const result1Promise = getData();

  // 第二次调用 getData() 会直接返回 dataGlobalPromise 中已有的 Promise
  const result2Promise = getData();

  // 第三次调用 getData() 同样直接返回已有的 Promise
  const result3Promise = getData();

  // 等待所有 Promise 解析
  const [result1, result2, result3] = await Promise.all([
    result1Promise,
    result2Promise,
    result3Promise
  ]);

  console.log("Result #1:", result1);
  console.log("Result #2:", result2);
  console.log("Result #3:", result3);

  console.log("\nConfirming dataGlobalPromise content:", dataGlobalPromise);

  // 再次调用 getData,验证 GetCoverage 不会再次执行
  console.log("\nCalling getData again to verify no new GetCoverage execution...");
  const result4 = await getData();
  console.log("Result #4:", result4);

})();

代码解析:

  1. let dataGlobalPromise;:声明一个变量,初始值为 undefined。
  2. const getData = () => (dataGlobalPromise ??= GetCoverage("all"));:这是核心所在。
    • 首次调用 getData() 时:dataGlobalPromise 是 undefined,满足 ??= 的条件。GetCoverage("all") 会被执行,返回一个 Promise 对象。这个 Promise 对象随后被赋值给 dataGlobalPromise,并作为 getData 函数的返回值。此时,控制台会输出 Executing GetCoverage for: all。
    • 后续调用 getData() 时:dataGlobalPromise 已经包含了之前赋值的 Promise 对象(无论它是否已解决)。此时 dataGlobalPromise 不再是空值,因此 ??= 运算符的右侧表达式 GetCoverage("all") 不会被执行。getData 函数直接返回 dataGlobalPromise 中已存在的 Promise 对象。控制台不会再次输出 Executing GetCoverage for: all。

通过这种方式,无论 getData 被调用多少次,GetCoverage 函数(模拟数据库查询)都只会被执行一次。所有对 getData 的调用都会返回同一个 Promise 对象,它们最终都会解析出相同的数据。

优势与注意事项

优势

  • 性能提升:显著减少了不必要的数据库或API查询,降低了服务器负载,提高了应用程序的响应速度。
  • 代码简洁:使用 ??= 运算符使缓存逻辑变得非常紧凑和易读。
  • 并发处理:即使多个函数同时请求数据,它们也会共享同一个 Promise,避免了竞态条件和重复工作。

注意事项

  • 错误处理:如果 GetCoverage 函数返回的 Promise 被拒绝(例如,数据库连接失败),那么 dataGlobalPromise 将会是一个被拒绝的 Promise。所有后续调用 getData().then(...) 的 catch 块都将捕获到这个错误。这是期望的行为,但需要确保你的代码能够正确处理这种情况。
  • 数据时效性:这种方法实现了永久缓存,除非应用程序重启或手动清除 dataGlobalPromise。如果数据需要定期刷新或根据某些条件失效,你需要额外的逻辑来重置 dataGlobalPromise(例如,将其设为 undefined 或 null),以便下次调用 getData 时能重新获取数据。
  • 作用域:dataGlobalPromise 变量的声明位置决定了缓存的作用域。在模块顶级声明它,可以使其在整个模块生命周期内有效。如果希望缓存仅在特定组件或类实例中有效,则应将其作为组件状态或类属性进行管理。

总结

通过巧妙地结合JavaScript的空值合并赋值运算符 (??=) 和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语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

238

2023.09.22

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

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

519

2024.03.01

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

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

1505

2023.10.24

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

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

233

2024.02.23

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

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

107

2025.10.17

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

532

2023.09.20

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

82

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

97

2025.09.18

AO3官网入口与中文阅读设置 AO3网页版使用与访问
AO3官网入口与中文阅读设置 AO3网页版使用与访问

本专题围绕 Archive of Our Own(AO3)官网入口展开,系统整理 AO3 最新可用官网地址、网页版访问方式、正确打开链接的方法,并详细讲解 AO3 中文界面设置、阅读语言切换及基础使用流程,帮助用户稳定访问 AO3 官网,高效完成中文阅读与作品浏览。

89

2026.02.02

热门下载

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

精品课程

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

共58课时 | 4.5万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.7万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.2万人学习

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

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