JavaScript Promise 的立即执行问题

今天在练习高并发任务,发现我对于Promise的执行时机有所误解,需求是构建一个任务数组用于后续按需执行。

问题代码

js 复制代码
    const mockUserDataTask = (id) =>
      new Promise((resolve) =>{
        console.log(`用户数据 ${id} 加载中`),
        setTimeout(() => resolve(`用户数据 ${id} 加载完成`), 3000)
      });

		 const test = [
		  { id: "task-1", func: mockUserDataTask(1) },
		  { id: "task-2", func: mockUserDataTask(2) },
		  { id: "task-3", func: mockUserDataTask(3) },
		  { id: "task-4", func: mockUserDataTask(4) },
		];

    test[0].func.then((res) => {
      console.log(res);
      test[2].func.then((res) => {
        console.log(res);
      });
    });

    test[1].func.then((res) => {
      console.log(res);
      test[3].func.then((res) => {
        console.log(res);
      });
    });

预期执行顺序:

先启动任务1和任务2,完成后启动任务3和任务4

实际执行顺序:

所有4个任务同时启动


为什么会同步加载4个Promise 对象?

Promise 对象在创建时会立即执行。

💡 这是因为Promise的构造函数部分是同步执行的,传入的 executor 函数会立即调用;而resolve/reject回调的(.then()/.catch())是异步的属于Microtask(微任务)

  • 当我直接调用 mockUserDataTask(id) 时,Promise 构造函数会立即执行
  • 这意味着我在创建 test 数组时,4个 Promise 已经同时开始执行(包括它们的 setTimeout
  • 所以会立即看到所有4个"加载中"

采用闭包的方式解决这一问题:

  • 外层函数返回的是一个新的函数,而不是直接返回Promise
  • 只有在调用 func() 时才会创建并执行Promise

正确写法

// ❌ Promise立即执行 const task = new Promise(executor);

// ✅ 延迟执行(闭包) const createTask = () => new Promise(executor);

js 复制代码
	  //使用闭包有效解决
	  const mockUserDataTask = (id) => () => new Promise((resolve) => {
      console.log(`用户数据 ${id} 加载中`)
      setTimeout(() => {
          console.log('执行了')
          resolve(`用户数据 ${id} 加载完成`)
      }, 3000)
    });
    
    test[0].func().then((res) => {
      console.log(res);
      test[2].func().then((res) => {
        console.log(res);
      });
    });
相关推荐
NoneCoder1 分钟前
正则表达式与文本处理的艺术
前端·javascript·面试·正则表达式
海盐泡泡龟2 小时前
Javascript本地存储的方式有哪些?区别及应用场景?(含Deep Seek讲解)
开发语言·javascript·ecmascript
Elastic 中国社区官方博客4 小时前
JavaScript 中使用 Elasticsearch 的正确方式,第一部分
大数据·开发语言·javascript·数据库·elasticsearch·搜索引擎·全文检索
万物得其道者成4 小时前
从零开始创建一个 Next.js 项目并实现一个 TodoList 示例
开发语言·javascript·ecmascript
sunbyte5 小时前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Expanding Cards (展开式卡片)
javascript·vue.js·ecmascript
肠胃炎5 小时前
React Contxt详解
javascript·react.js·ecmascript
xx24065 小时前
React Native简介
javascript·react native·react.js
重生之后端学习6 小时前
02-前端Web开发(JS+Vue+Ajax)
java·开发语言·前端·javascript·vue.js
布鲁斯的快乐小屋6 小时前
axios的基本使用
javascript·ajax
来自星星的坤9 小时前
【Vue 3 + Vue Router 4】如何正确重置路由实例(resetRouter)——避免“VueRouter is not defined”错误
前端·javascript·vue.js