Promise扁平化嵌套靠.then()返回新Promise而非嵌套调用;串行需显式return并传递参数;Promise.all/race解决并行非串行;避免在.then中新建Promise或包裹同步逻辑。
Promise 本身不“解决”回调地狱,它提供了一种链式写法让嵌套变平——关键在 .then() 返回新 Promise,而不是在上一个 .then() 里再写 .then()。
常见错误是仍把异步操作塞进回调函数体里,比如:
fetch('/api/a')
.then(res => res.json())
.then(data => {
fetch('/api/b?x=' + data.id) // ❌ 忘记 return,这里返回的是 undefined
.then(r => r.json())
.then(d => console.log(d));
});
正确做法是显式 return 下一个 Promise:
.then() 回调里,若要继续链式调用,必须 return 一个 Promise(或可被 Promise.resolve() 包装的值).then() 会立即收到 undefined
throw new Error())会被自动捕获进下一个 .catch(),不用手动 reject()
串行即“等前一个完成再发起下一个”,典型场景是分页拉取、依赖上一步结果的 API 调用。直接用 async/await 最直观,但纯 Promise 也能做到:
Promise.resolve()
.then(() => fetch('/api/user'))
.then(res => res.json())
.then(user => fetch('/api/posts?uid=' + user.id))
.then(res => res.json())
.then(posts => console.log(posts))
.catch(err => console.error(err));
注意点:
Promise.resolve() 统一入口,避免第一个 fetch 出错时无法被最外层 .catch() 捕获fetch().then(() => fetch()),这会丢失第一个请求的响应数据;应靠参数传递(如上例中 user)Promise.all([p1, p2])
这两个方法常被拿来“替代回调地狱”,但它们解决的是不同问题:Promise.all 是并行+全成功,Promise.race 是竞速,都不等于“链式流程控制”。
典型误用:
Promise.all([fetch('/a'), fetch('/b')]) 去处理有依赖关系的任务(比如 b 需要 a 的返回 id)→ 应该串行,不是并行Promise.race([timeout(), apiCall()]) 但没给 timeout() 加 reject → race 只取第一个 settle 结果,如果 timeout 永不 reject,就会永远卡住Promise.all 中任
Promise.allSettled(注意兼容性)常见原因不是 Promise 不好用,而是没切断旧习惯:
.then() 里又写一层 new Promise((resolve, reject) => {...}) → 其实多数时候可以直接 return fetch(...)
setTimeout 或额外 Promise → 同步代码不需要 Promise 包裹.then(),比如每个字段提取都单独写一行 .then(x => x.data).then(y => y.items),反而增加阅读负担真正难的不是语法,是区分哪些操作天然异步(API、定时器、文件读取)、哪些只是同步计算——后者不该进 Promise 链。
来电咨询